null
, undefined
类型默认情况下null
和 undefined
是所有类型的子类型, 即可以把null
和 undefined
赋值给其他类型。
注意:
tsconfig.json指定了"strictNullChecks":true
, null
和 undefined
只能赋值给 any
、 unknown
和它们各自的类型, undefined
可以赋值给void
类型;
以下代码在严格模式下("strictNullChecks":true
)下运行:
let v: void = undefined; // 没毛病
let v1: void = null; // Type 'null' is not assignable to type 'void'.
let notSure: unknown = 'unknown';
let a: any = 18;
let n: null = null;
let u: undefined = undefined;
a = n; // 没毛病
a = u; // 没毛病
notSure = n; // 没毛病
notSure = u; // 没毛病
n = u; // Type 'undefined' is not assignable to type 'null'.
以下代码在非严格模式下(指定了"strictNullChecks":false
), 可以把null
和 undefined
赋值给其他类型
// null和undefined赋值给number
let num: number = 18;
num = null;
num = undefined;
// null和undefined赋值给string
let str: string = 'hello';
str = null;
str = undefined;
// null和undefined赋值给boolean
let bool: boolean = false;
bool = null;
bool = undefined;
// null和undefined赋值给bigint
let big: bigint = 10n;
big = null;
big = undefined;
// null和undefined赋值给symbol
let sym: symbol = Symbol('foo');
sym = null;
sym = undefined;
// null和undefined赋值给object
let obj: object = {};
obj = null;
obj = undefined;
注意:
如果出现BigInt literals are not available when targeting lower than ES2020.
需要把ypeScript编译器目标设置为es2020
number、string、symbol、boolean、bigint
原始类型在赋值过程中改变类型是不被允许的。
虽然number
和 bigint
都表示数字,但是这两个类型不兼容。
let num: number = 100;
num = 'hello'; // Type 'string' is not assignable to type 'number'.
let str: string = 'xman';
str = symbol('1'); // Type 'symbol' is not assignable to type 'string'.
let sym: symbol = Symbol('foo');
sym = false; // Type 'boolean' is not assignable to type 'symbol'.
let big: bigint = 10n;
big = 1; // Type 'number' is not assignable to type 'bigint'.
let obj: object = {};
obj = 'xxx'; // Type 'string' is not assignable to type 'object'.
Number、String、Symbol、Boolean、Bigint
包装类型注意: Number、String、Symbol、Boolean、Bigint
是包装类型, 而number、string、symbol、boolean、bigint
是原始类型,不要混淆。
从类型兼容性上看, 原始类型兼容对应的对象包装类型, 对象类型不兼容对应的原始类型。
如:
let str: string = 'hello';
let Str: String = String('hello');
str = Str;
// Type 'String' is not assignable to type 'string'.
// 'string' is a primitive, but 'String' is a wrapper object. Prefer using 'string' when possible.
Str = 'hello'; // 没毛病
在TypeScript中, 任何类型都可以被归为 any
类型,这让 any 类型成为了类型系统的顶级类型.
let a: any = 'any';
let notSure: unknown = 'unknown';
let v: void = undefined;
let ne: never;
// 以下都没毛病
a = ne;
v = a;
a = v;
a = notSure;
a = 100;
a = 10n;
a = false;
a = undefined;
a = null;
a = [];
a = {};
any
类型。let a: any = 'anything';
console.log(a.name);
console.log(a.info.address);
a.setName('xman');
a.getName();
any
类型 而完全不被类型检查:let a; // 推断为any类型
a = 'hello';
a = false;
a = 100;
a = undefined;
a = null;
a = {};
最后总结下: 使用 any,我们将会失去通常由 TypeScript 的静态类型系统所给予的所有保护。因此,如果我们无法使用更具体的类型或 unknown,则只能将其用作最后的手段。 一定要慎用any
类型。
unknown
与any
都是TypeScript中的顶级类型。 所以所有类型都可以分配给unknown
:
let value: unknown;
value = true;
value = 42;
value = "Hello World";
value = [];
value = {};
value = Math.random;
value = null;
value = undefined;
value = new TypeError();
value = Symbol("type");
unknown
类型只能被赋值给 any
类型和 unknown
类型本身。
let value: unknown;
let value1: unknown = value; // 没毛病
let a: any = value; // 没毛病
let num: number = value; // Type 'unknown' is not assignable to type 'number'.
TypeScript 不允许我们对类型为 unknown 的值执行任意操作。相反,我们必须首先执行某种类型检查以缩小我们正在使用的值的类型范围。
function foo (value: unknown) {
let val = value.toFixed(2); // Property 'toFixed' does not exist on type 'unknown'.
}
我们可以使用如下方法进行缩小未知范围:
function foo (value: unknown) {
let val = (value as number).toFixed(2);
}
function foo (value: unknown) {
if (typeof value === 'number') {
let val = value.toFixed(2);
}
}
const isNumber = (val: unknown): val is number => typeof val === 'number';
function foo (value: unknown) {
if (isNumber(value)) {
let val = value.toFixed(2);
}
}
never
类型表示的是那些永不存在的值的类型
never类型同null和undefined一样,也是任何类型的子类型,也可以赋值给任何类型, 但是没有类型是never的子类型或者可以赋值给never类型(除了never本身之外),即使any也不可以赋值给never。
never 类型是 TypeScript 中的底层类型。它自然被分配的一些例子:
let ne = (() => { while(true) {} })(); // ne 为never类型
let ne = (() => { throw new Error("异常"); })(); // ne 为never类型
let ne: never;
let ne1: never;
let num: number = 1;
num = ne; // 没毛病
ne = 100; // Type 'number' is not assignable to type 'never'.
ne = ne1; // 没毛病
JavaScript中的void
JavaScript中的void
是一个运算符, 对给定的表达式进行求值,然后返回 undefined。具体可看MDN void
let i = void 100;
i === undefined; // true
我们为什么需要这样的东西?首先在早期,人们能够覆盖 undefined 并给它一个实际值。 void 总是返回 real undefined。
其次,这是一种调用立即调用函数的好方法:
void function() {
console.log('What')
}()
TypeScript中的void
TypeScript 中的 void 是 undefined 的子类型。 JavaScript 中的函数总是返回一些东西。要么它是一个值,要么是 undefined.
因为没有返回值的函数总是返回undefined,下面函数返回数据类型定义为void
类型,告诉开发人员这个函数返回 undefined:
declare function iTakeNoParameters (x: number): void;
void用作参数声明
declare function iTakeNoParameters (x: void): void;
iTakeNoParameters();
iTakeNoParameters(undefined);
iTakeNoParameters(void 100);
那void
和undefined
有什么区别呢?
区别在于 作为返回类型的void
可以用不同的类型替换,以允许高级回调模式:
function doSomething (callback: () => void) {
let c = callback(); // 此时c为void类型
// ...
}
// 返回number类型
function aNumberCallback (): number {
return 2;
}
doSomething(aNumberCallback);
如果你想确保传递只返回 undefined 的函数(如“nothing”),请确保调整你的回调方法签名:
function doSomething (callback: () => undefined) {
let c = callback(); // 此时c为void类型
// ...
}
// 返回number类型
function aNumberCallback (): number {
return 2;
}
doSomething(aNumberCallback);
// Argument of type '() => number' is not assignable to parameter of type '() => undefined'.
// Type 'number' is not assignable to type 'undefined'.
此外, 函数如果没有显式的返回undefined的,我们需要定义成void类型,而不是undefined类型。否则将报错。
function foo (): void {
console.log('---foo----');
}
// A function whose declared type is neither 'void' nor 'any' must return a value.
function foo1(): undefined {
console.log('---foo1----');
}
foo1();
// 没毛病
function foo2 (): undefined {
return undefined;
}
以上ts代码均在 https://www.typescriptlang.org/play 上运行过,版本为4.7.2。
最后, 如有错误,欢迎各位大佬指点!感谢!
https://fettblog.eu/void-in-javascript-and-typescript/
https://juejin.cn/post/6844903866073350151