最佳实践
常规类型
Number
,String
,Boolean
,Symbol
和Object
不要使用以下类型Number
,String
,Boolean
,Symbol
或Object
。
这些类型表示是非原始的封箱后的对象类型,它们几乎没有在 JavaScript 代码里被正确地使用过。
/* 错误 */
function reverse(s: String): String;
应该使用number
,string
,boolean
和symbol
类型。
/* 正确 */
function reverse(s: string): string;
使用非原始的object
类型来代替Object
类型
泛型
不要定义没有使用过类型参数的泛型类型。 更多详情请参考:TypeScript FAQ page。
any
请尽量不要使用any
类型,除非你正在将 JavaScript 代码迁移到 TypeScript 代码。
编译器实际上会将any
视作“对其关闭类型检查”。
使用它与在每个变量前使用@ts-ignore
注释是一样的。
它只在首次将 JavaScript 工程迁移到 TypeScript 工程时有用,因为你可以把还没有迁移完的实体标记为any
类型,但在完整的 TypeScript 工程中,这样做就会禁用掉类型检查。
如果你不清楚要接收什么类型的数据,或者你希望接收任意类型并直接向下传递而不使用它,那么就可以使用unknown
类型。
回调函数类型
回调函数的返回值类型
不要为返回值会被忽略的回调函数设置返回值类型any
:
/* 错误 */
function fn(x: () => any) {
x();
}
应该为返回值会被忽略的回调函数设置返回值类型void
:
/* 正确 */
function fn(x: () => void) {
x();
}
原因:使用void
相对安全,因为它能防止不小心使用了未经检查的x
的返回值:
function fn(x: () => void) {
var k = x(); // oops! meant to do something else
k.doSomething(); // error, but would be OK if the return type had been 'any'
}
回调函数里的可选参数
不要在回调函数里使用可选参数,除非这是你想要的:
/* 错误 */
interface Fetcher {
getObject(done: (data: any, elapsedTime?: number) => void): void;
}
这里有具体的意义:done
回调函数可以用 1 个参数或 2 个参数调用。
代码的大意是说该回调函数不关注是否有elapsedTime
参数, 但是不需要把这个参数定义为可选参数来达到此目的 --
因为总是允许提供一个接收较少参数的回调函数。
应该将回调函数定义为无可选参数:
/* 正确 */
interface Fetcher {
getObject(done: (data: any, elapsedTime: number) => void): void;
}
重载与回调函数
不要因回调函数的参数数量不同而编写不同的重载。
/* WRONG */
declare function beforeAll(action: () => void, timeout?: number): void;
declare function beforeAll(
action: (done: DoneFn) => void,
timeout?: number
): void;
应该只为最大数量参数的情况编写一个重载:
/* 正确 */
declare function beforeAll(
action: (done: DoneFn) => void,
timeout?: number
): void;
原因:回调函数总是允许忽略某个参数的,因此没必要为缺少可选参数的情况编写重载。 为缺少可选参数的情况提供重载可能会导致类型错误的回调函数被传入,因为它会匹配到第一个重载。
函数重载
顺序
不要把模糊的重载放在具体的重载前面:
/* 错误 */
declare function fn(x: any): any;
declare function fn(x: HTMLElement): number;
declare function fn(x: HTMLDivElement): string;
var myElem: HTMLDivElement;
var x = fn(myElem); // x: any, wat?
应该将重载排序,把具体的排在模糊的之前:
/* 正确 */
declare function fn(x: HTMLDivElement): string;
declare function fn(x: HTMLElement): number;
declare function fn(x: any): any;
var myElem: HTMLDivElement;
var x = fn(myElem); // x: string, :)
原因:当解析函数调用的时候,TypeScript 会选择匹配到的第一个重载。 当位于前面的重载比后面的“更模糊”,那么后面的会被隐藏且不会被选用。
使用可选参数
不要因为只有末尾参数不同而编写不同的重载:
/* WRONG */
interface Example {
diff(one: string): number;
diff(one: string, two: string): number;
diff(one: string, two: string, three: boolean): number;
}
应该尽可能使用可选参数:
/* OK */
interface Example {
diff(one: string, two?: string, three?: boolean): number;
}
注意,这只在返回值类型相同的情况是没问题的。
原因:有以下两个重要原因。
TypeScript 解析签名兼容性时会查看是否某个目标签名能够使用原参数调用, 且允许额外的参数。 下面的代码仅在签名被正确地使用可选参数定义时才会暴露出一个 bug:
function fn(x: (a: string, b: number, c: number) => void) {}
var x: Example;
// When written with overloads, OK -- used first overload
// When written with optionals, correctly an error
fn(x.diff);
第二个原因是当使用了 TypeScript “严格检查 null” 的特性时。
因为未指定的参数在 JavaScript 里表示为undefined
,通常明确地为可选参数传入一个undefined
不会有问题。
这段代码在严格 null
模式下可以工作:
var x: Example;
// When written with overloads, incorrectly an error because of passing 'undefined' to 'string'
// When written with optionals, correctly OK
x.diff("something", true ? undefined : "hour");
使用联合类型
不要仅因某个特定位置上的参数类型不同而定义重载:
/* 错误 */
interface Moment {
utcOffset(): number;
utcOffset(b: number): Moment;
utcOffset(b: string): Moment;
}
应该尽可能地使用联合类型:
/* 正确 */
interface Moment {
utcOffset(): number;
utcOffset(b: number | string): Moment;
}
注意,我们没有让b
成为可选的,因为签名的返回值类型不同。
原因:这对于那些为该函数传入了值的使用者来说很重要。
function fn(x: string): void;
function fn(x: number): void;
function fn(x: number | string) {
// When written with separate overloads, incorrectly an error
// When written with union types, correctly OK
return moment().utcOffset(x);
}