以下列表概述了在 JavaScript 文件中使用 JSDoc 注释提供类型信息时当前支持的构造。
注意,任何未在下面明确列出的标签(例如 @async
)目前尚不支持。
类型
类
- 属性修饰符
@public
,@private
,@protected
,@readonly
@override
@extends
(或@augments
)@implements
@class
(或@constructor
)@this
文档
文档标签在 TypeScript 和 JavaScript 中都有效。
其他
其含义通常与 jsdoc.app 中给出的标签含义相同或为其超集。以下代码描述了差异,并提供了一些每个标签的示例用法。
注意:您可以使用 游乐场来探索 JSDoc 支持。
类型
@type
您可以使用“@type”标签引用类型。该类型可以是
- 原始类型,例如
string
或number
。 - 在 TypeScript 声明中声明,可以是全局的,也可以是导入的。
- 在 JSDoc
@typedef
标签中声明。
您可以使用大多数 JSDoc 类型语法和任何 TypeScript 语法,从 最基本的,例如string
到 最先进的,例如条件类型。
jsTry
/*** @type {string}*/vars ;/** @type {Window} */varwin ;/** @type {PromiseLike<string>} */varpromisedString ;// You can specify an HTML Element with DOM properties/** @type {HTMLElement} */varmyElement =document .querySelector (selector );element .dataset .myData = "";
@type
可以指定联合类型 - 例如,某事物可以是字符串或布尔值。
jsTry
/*** @type {string | boolean}*/varsb ;
您可以使用各种语法来指定数组类型
jsTry
/** @type {number[]} */varns ;/** @type {Array.<number>} */varjsdoc ;/** @type {Array<number>} */varnas ;
您还可以指定对象字面量类型。例如,具有属性“a”(字符串)和“b”(数字)的对象使用以下语法
jsTry
/** @type {{ a: string, b: number }} */varvar9 ;
您可以使用字符串和数字索引签名来指定类似于映射和数组的对象,可以使用标准 JSDoc 语法或 TypeScript 语法。
jsTry
/*** A map-like object that maps arbitrary `string` properties to `number`s.** @type {Object.<string, number>}*/varstringToNumber ;/** @type {Object.<number, object>} */vararrayLike ;
前面的两种类型等效于 TypeScript 类型{ [x: string]: number }
和{ [x: number]: any }
。编译器理解这两种语法。
您可以使用 TypeScript 或 Google Closure 语法来指定函数类型
jsTry
/** @type {function(string, boolean): number} Closure syntax */varsbn ;/** @type {(s: string, b: boolean) => number} TypeScript syntax */varsbn2 ;
或者您也可以使用未指定的Function
类型
jsTry
/** @type {Function} */varfn7 ;/** @type {function} */varfn6 ;
Closure 中的其他类型也适用
jsTry
/*** @type {*} - can be 'any' type*/varstar ;/*** @type {?} - unknown type (same as 'any')*/varquestion ;
强制转换
TypeScript 从 Google Closure 借用了强制转换语法。这使您可以通过在任何带括号的表达式之前添加@type
标签将类型强制转换为其他类型。
jsTry
/*** @type {number | string}*/varnumberOrString =Math .random () < 0.5 ? "hello" : 100;vartypeAssertedNumber = /** @type {number} */ (numberOrString );
您甚至可以像 TypeScript 一样强制转换为const
jsTry
letone = /** @type {const} */(1);
导入类型
您可以使用导入类型从其他文件导入声明。此语法是 TypeScript 特定的,与 JSDoc 标准不同
jsTry
// @filename: types.d.tsexport typePet = {name : string,};// @filename: main.js/*** @param {import("./types").Pet} p*/functionwalk (p ) {console .log (`Walking ${p .name }...`);}
导入类型可以在类型别名声明中使用
jsTry
/*** @typedef {import("./types").Pet} Pet*//*** @type {Pet}*/varmyPet ;myPet .name ;
如果不知道类型,或者类型很大且难以键入,可以使用导入类型来获取模块中值的类型
jsTry
/*** @type {typeof import("./accounts").userAccount}*/varx =require ("./accounts").userAccount ;
@param
和 @returns
@param
使用与 @type
相同的类型语法,但添加了参数名称。参数也可以通过将名称用方括号括起来来声明为可选的
jsTry
// Parameters may be declared in a variety of syntactic forms/*** @param {string} p1 - A string param.* @param {string=} p2 - An optional param (Google Closure syntax)* @param {string} [p3] - Another optional param (JSDoc syntax).* @param {string} [p4="test"] - An optional param with a default value* @returns {string} This is the result*/functionstringsStringStrings (p1 ,p2 ,p3 ,p4 ) {// TODO}
同样,对于函数的返回类型
jsTry
/*** @return {PromiseLike<string>}*/functionps () {}/*** @returns {{ a: string, b: number }} - May use '@returns' as well as '@return'*/functionab () {}
@typedef
、@callback
和 @param
您可以使用 @typedef
定义复杂类型。类似的语法适用于 @param
。
jsTry
/*** @typedef {Object} SpecialType - creates a new type named 'SpecialType'* @property {string} prop1 - a string property of SpecialType* @property {number} prop2 - a number property of SpecialType* @property {number=} prop3 - an optional number property of SpecialType* @prop {number} [prop4] - an optional number property of SpecialType* @prop {number} [prop5=42] - an optional number property of SpecialType with default*//** @type {SpecialType} */varspecialTypeObject ;specialTypeObject .prop3 ;
您可以在第一行使用 object
或 Object
。
jsTry
/*** @typedef {object} SpecialType1 - creates a new type named 'SpecialType1'* @property {string} prop1 - a string property of SpecialType1* @property {number} prop2 - a number property of SpecialType1* @property {number=} prop3 - an optional number property of SpecialType1*//** @type {SpecialType1} */varspecialTypeObject1 ;
@param
允许类似的语法用于一次性类型规范。请注意,嵌套属性名称必须以参数的名称为前缀
jsTry
/*** @param {Object} options - The shape is the same as SpecialType above* @param {string} options.prop1* @param {number} options.prop2* @param {number=} options.prop3* @param {number} [options.prop4]* @param {number} [options.prop5=42]*/functionspecial (options ) {return (options .prop4 || 1001) +options .prop5 ;}
@callback
类似于 @typedef
,但它指定函数类型而不是对象类型
jsTry
/*** @callback Predicate* @param {string} data* @param {number} [index]* @returns {boolean}*//** @type {Predicate} */constok = (s ) => !(s .length % 2);
当然,所有这些类型都可以使用 TypeScript 语法在一行 @typedef
中声明
js
/** @typedef {{ prop1: string, prop2: string, prop3?: number }} SpecialType *//** @typedef {(data: string, index?: number) => boolean} Predicate */
@template
您可以使用 @template
标签声明类型参数。这使您可以创建泛型的函数、类或类型
jsTry
/*** @template T* @param {T} x - A generic parameter that flows through to the return type* @returns {T}*/functionid (x ) {returnx ;}consta =id ("string");constb =id (123);constc =id ({});
使用逗号或多个标签来声明多个类型参数
js
/*** @template T,U,V* @template W,X*/
您也可以在类型参数名称之前指定类型约束。列表中只有第一个类型参数受到约束
jsTry
/*** @template {string} K - K must be a string or string literal* @template {{ serious(): string }} Seriousalizable - must have a serious method* @param {K} key* @param {Seriousalizable} object*/functionseriousalize (key ,object ) {// ????}
最后,您可以为类型参数指定默认值
jsTry
/** @template [T=object] */classCache {/** @param {T} initial */constructor(initial ) {}}letc = newCache ()
@satisfies
@satisfies
提供对 TypeScript 中的后缀 运算符 satisfies
的访问。Satisfies 用于声明一个值实现了一个类型,但不会影响值的类型。
jsTry
// @ts-check/*** @typedef {"hello world" | "Hello, world"} WelcomeMessage*//** @satisfies {WelcomeMessage} */constmessage = "hello world"/** @satisfies {Type '"Hello world!"' does not satisfy the expected type 'WelcomeMessage'.1360Type '"Hello world!"' does not satisfy the expected type 'WelcomeMessage'.WelcomeMessage } */constfailingMessage = "Hello world!"/** @type {WelcomeMessage} */constmessageUsingType = "hello world"
类
类可以声明为 ES6 类。
jsTry
classC {/*** @param {number} data*/constructor(data ) {// property types can be inferredthis.name = "foo";// or set explicitly/** @type {string | null} */this.title = null;// or simply annotated, if they're set elsewhere/** @type {number} */this.size ;this.initialize (data ); // Should error, initializer expects a string}/*** @param {string} s*/initialize = function (s ) {this.size =s .length ;};}varc = newC (0);// C should only be called with new, but// because it is JavaScript, this is allowed and// considered an 'any'.varresult =C (1);
它们也可以声明为构造函数;使用 @constructor
以及 @this
来表示 this。
属性修饰符
@public
、@private
和 @protected
的工作方式与 TypeScript 中的 public
、private
和 protected
完全相同
jsTry
// @ts-checkclassCar {constructor() {/** @private */this.identifier = 100;}printIdentifier () {console .log (this.identifier );}}constc = newCar ();Property 'identifier' is private and only accessible within class 'Car'.2341Property 'identifier' is private and only accessible within class 'Car'.console .log (c .); identifier
@public
始终隐含,可以省略,但表示属性可以从任何地方访问。@private
表示属性只能在包含的类中使用。@protected
表示属性只能在包含的类及其所有派生子类中使用,但不能在包含类的不同实例上使用。
@public
、@private
和 @protected
在构造函数中不起作用。
@readonly
@readonly
修饰符确保属性仅在初始化期间写入。
jsTry
// @ts-checkclassCar {constructor() {/** @readonly */this.identifier = 100;}printIdentifier () {console .log (this.identifier );}}constc = newCar ();console .log (c .identifier );
@override
@override
的工作方式与 TypeScript 中相同;将其用于覆盖基类方法的方法。
jsTry
export classC {m () { }}classD extendsC {/** @override */m () { }}
在 tsconfig 中设置 noImplicitOverride: true
以检查覆盖。
@extends
当 JavaScript 类扩展泛型基类时,没有 JavaScript 语法用于传递类型参数。@extends
标签允许这样做。
jsTry
/*** @template T* @extends {Set<T>}*/classSortableSet extendsSet {// ...}
请注意,@extends
仅适用于类。目前,构造函数无法扩展类。
@implements
同样,没有 JavaScript 语法用于实现 TypeScript 接口。@implements
标签的工作方式与 TypeScript 中相同。
jsTry
/** @implements {Print} */classTextBook {// TODO}}
@constructor
编译器根据 this 属性赋值推断构造函数,但如果添加 @constructor
标签,可以使检查更严格,建议更好。
jsTry
/*** @constructor* @param {number} data*/functionC (data ) {// property types can be inferredthis.name = "foo";// or set explicitly/** @type {string | null} */this.title = null;// or simply annotated, if they're set elsewhere/** @type {number} */this.size ;this.Argument of type 'number' is not assignable to parameter of type 'string'.2345Argument of type 'number' is not assignable to parameter of type 'string'.initialize (); data }/*** @param {string} s*/C .prototype .initialize = function (s ) {this.size =s .length ;};varc = newC (0);c .size ;varValue of type 'typeof C' is not callable. Did you mean to include 'new'?2348Value of type 'typeof C' is not callable. Did you mean to include 'new'?result =C (1);
使用 @constructor
,this
在构造函数 C
内部进行检查,因此您将获得 initialize
方法的建议,并且如果传递数字,则会收到错误。您的编辑器也可能会在您调用 C
而不是构造它时显示警告。
不幸的是,这意味着既可调用又可作为构造函数的函数不能使用 @constructor
。
@this
编译器通常可以在有上下文的情况下弄清楚 this
的类型。当它没有时,您可以使用 @this
显式指定 this
的类型。
jsTry
/*** @this {HTMLElement}* @param {*} e*/functioncallbackForLater (e ) {this.clientHeight =parseInt (e ); // should be fine!}
文档
@deprecated
当一个函数、方法或属性被弃用时,你可以通过在它上面添加 /** @deprecated */
JSDoc 注释来让用户知道。该信息会在代码补全列表中显示,并且作为编辑器可以特殊处理的建议诊断信息。在像 VS Code 这样的编辑器中,弃用的值通常以删除线样式显示 像这样。
jsTry
/** @deprecated */constapiV1 = {};constapiV2 = {};apiV ;
@see
@see
允许你链接到程序中的其他名称
tsTry
typeBox <T > = {t :T }/** @see Box for implementation details */typeBoxify <T > = { [K in keyofT ]:Box <T > };
一些编辑器会将 Box
变成一个链接,方便你跳转到它并返回。
@link
@link
与 @see
相似,但它可以用于其他标签中
tsTry
typeBox <T > = {t :T }/** @returns A {@link Box} containing the parameter. */functionbox <U >(u :U ):Box <U > {return {t :u };}
其他
@enum
@enum
标签允许你创建一个对象字面量,其成员都是指定类型的。与 JavaScript 中的大多数对象字面量不同,它不允许其他成员。@enum
旨在与 Google Closure 的 @enum
标签兼容。
jsTry
/** @enum {number} */constJSDocState = {BeginningOfLine : 0,SawAsterisk : 1,SavingComments : 2,};JSDocState .SawAsterisk ;
请注意,@enum
与 TypeScript 的 enum
非常不同,并且比它简单得多。但是,与 TypeScript 的枚举不同,@enum
可以具有任何类型
jsTry
/** @enum {function(number): number} */constMathFuncs = {add1 : (n ) =>n + 1,id : (n ) => -n ,sub1 : (n ) =>n - 1,};MathFuncs .add1 ;
@author
你可以使用 @author
指定项目的作者
tsTry
/*** Welcome to awesome.ts*/
请记住用尖括号包围电子邮件地址。否则,@example
将被解析为一个新的标签。
其他支持的模式
jsTry
varsomeObj = {/*** @param {string} param1 - JSDocs on property assignments work*/x : function (param1 ) {},};/*** As do jsdocs on variable assignments* @return {Window}*/letsomeFunc = function () {};/*** And class methods* @param {string} greeting The greeting to use*/Foo .prototype .sayHi = (greeting ) =>console .log ("Hi!");/*** And arrow function expressions* @param {number} x - A multiplier*/letmyArrow = (x ) =>x *x ;/*** Which means it works for function components in JSX too* @param {{a: string, b: number}} props - Some param*/varfc = (props ) => <div >{props .a .charAt (0)}</div >;/*** A parameter can be a class constructor, using Google Closure syntax.** @param {{new(...args: any[]): object}} C - The class to register*/functionregisterClass (C ) {}/*** @param {...string} p1 - A 'rest' arg (array) of strings. (treated as 'any')*/functionfn10 (p1 ) {}/*** @param {...string} p1 - A 'rest' arg (array) of strings. (treated as 'any')*/functionfn9 (p1 ) {returnp1 .join ();}
不支持的模式
对象文字类型中属性类型上的后缀等于号不会指定可选属性
jsTry
/*** @type {{ a: string, b: number= }}*/varwrong ;/*** Use postfix question on the property name instead:* @type {{ a: string, b?: number }}*/varright ;
可空类型只有在 strictNullChecks
启用时才有意义
jsTry
/*** @type {?number}* With strictNullChecks: true -- number | null* With strictNullChecks: false -- number*/varnullable ;
TypeScript 原生语法是联合类型
jsTry
/*** @type {number | null}* With strictNullChecks: true -- number | null* With strictNullChecks: false -- number*/varunionNullable ;
非空类型没有意义,并且被视为其原始类型
jsTry
/*** @type {!number}* Just has type number*/varnormal ;
与 JSDoc 的类型系统不同,TypeScript 只允许您将类型标记为包含 null 或不包含 null。没有明确的非空性 - 如果 strictNullChecks 启用,则 number
不可空。如果它被禁用,则 number
可空。
不支持的标签
TypeScript 会忽略任何不支持的 JSDoc 标签。
以下标签存在支持它们的未解决问题