接口的兼容性
- 如果传入的变量和声明的类型不匹配,TS就会进行兼容性检查
- 原理是
Duck-Check,就是说只要目标类型中声明的属性变量在源类型中都存在就是兼容的``typescript interface Animal{ name:string; age:number; } interface Person { name:string; age:number; gender:number; } // 要判断目标类型Person是否能够兼容输入的源类型Animal` function getName(a:Animal):string{ return a.name; }
let p:Person = { name: ‘jack’, age: 10, gender: 0, } // Animal接口所需的属性,Person都有,所以getName的参数a也可以用Person接口 console.log(getName(p)); //jack
// 只有在传参的时候两个变量之间才会进行兼容性比较; // 注意:直接赋值的时候并不会进行兼容性比较。
<a name="VL6HK"></a>##### 直接赋值的时候并不会进行兼容性比较```typescriptinterface Animal{name:string;age:number;}interface Person {name:string;age:number;sex:string,}let p:Person = {name: 'jack', age: 10, sex: 'male'};// 情况1:报错let a1:Animal = {name: 'jack', age: 10, sex: 'male'};// 情况2:报错let a2:Animal;a2 = {name: 'jack', age: 10, sex: 'male'};// 情况3: 可以let a3:Animal = p;// 情况4: 可以let a4:Animal;a3 = p;
基本类型的兼容性
//基本数据类型也有兼容性判断let num:string|number;let str:string = 'jack'; //直接赋值的时候,不会进行兼容性比较num = str; //可以,间接赋值的时候,会进行兼容性比较,只要后者类型是前者类型的子类型,就可以。// str = num; //不可以//只要有toString()方法就可以赋给字符串变量let num2:{toString():string;}let str2:string = 'jack';num2 = str2; //可以// str2 = num2; //不可以,string上除了toString方法外,还有许多其他的方法,而这些是 num2 不具有的
类的兼容性
在TS中是结构类型系统,只会对比结构而不在意类型
class Animal{name!:string;}class Bird extends Animal {age!:number;}let a:Animal = new Bird(); //可以,Bird中具有Animal的name属性,所以Bird兼容Animal// let b:Bird = new Animal(); //报错,Animal中不具有Bird所需的age属性,所以Animal不兼容Bird
//如果父类和子类结构一样,也可以的class Animal{}class Bird extends Animal {}let a:Animal = new Bird(); //可以let b:Bird = new Animal(); //可以
//甚至没有关系的两个类的实例也是可以的class Animal{}class Book {}let a:Animal = new Book(); //可以let b:Book = new Animal(); //可以
函数的兼容性
比较函数的时候是要先比较函数的参数,再比较函数的返回值。
一切为了类型安全,为了使用的时候不报错。
比较参数
function add(x,y){// 我这边,只有2个参数// fn可以不用、用1个、用2个都行fn1();fn2(x);fn3(x,y);//但是,想要用 我都没有的,不行。fn4(x,y,z);}add(1,2);
type Func = (a:number, b:number) => void;let sum:Func;// 参数一样 兼容 可以赋值给sumfunction f1(a:number, b:number):void{}sum = f1;// 参数少一个,甚至无参数, 兼容function f2(a:number):void{}function f3():void{}sum = f2;sum = f3;// 参数多一个, 不兼容.//sum只能传2个参数,但是f4调用的时候需要3个参数,所以报错function f4(a:number, b:number, c:number):void{}sum = f4;sum(1,2);
比较返回值
// 这里只是类型定义,返回对象的话,不需要小括号包裹type GetPerson = () => {name:string, age:number};let getPerson:GetPerson;// 兼容,返回值一样可以function g1(){return {name:'jack', age: 18};}getPerson = g1;// 兼容,返回值多一个属性也可以function g2(){return {name:'jack', age: 18, gender: 0};}getPerson = g2;// 不兼容,返回值少一个属性可不行function g3(){return {name:'jack'};}getPerson = g3; //报错// 为什么这样设计,// 一切为了类型安全,为了使用的时候不报错。// 比如,有可能要调用返回值上的方法,如果没有age,age就是undefined,undefined.toFixed()就会发生错误getPerson().age.toFixed();
函数的协变与逆变
- 协变(Covariant):只在同一个方向。
- 逆变(Contravariant):只在相反的方向。
- 双向协变(Bivariant):包括同一个方向和不同方向。
不变(Invaraint):如果类型不完全相同,则它们是不兼容的。
A ≼ B 意味着 A 是 B 的子类型。
A → B 指的是以 A 为参数类型,以 B 为返回值类型的函数类型。
x : A 意味着 x 的类型为 A
返回值类型是协变的,而参数类型是逆变的
返回值类型可以传子类,参数可以传父类
- 参数逆变父类 返回值协变子类 搀你父,返鞋子
注意:在 TypeScript 中, 参数类型是双向协变的 ,也就是说既是协变又是逆变的,而这并不安全。但是现在你可以在 TypeScript 2.6 版本中通过
--strictFunctionTypes或--strict标记来修复这个问题
示例1
class Animal{}class Dog extends Animal{public name!:string;}class BlackDog extends Dog {public age!:number;}let animal:Animal = {};let dog:Dog = {name: '狗'};let blackDog:BlackDog = {name: '狗', age: 10};type Cb = (dog: Dog) => Dog; // 形参cb的约束类型function exec(cb: Cb){cb(dog); //示例1return cb(blackDog).name; //示例2};// 结论:// 函数cb的参数是逆变的,可以传父类;(传逆父)// 函数cb的返回值是协变的,可以传子类。(返协子)
实参cb的约束情况
// 参数是子类,返回值是子类 (报错)// 参数不安全,比如示例1中函数参数cb调用时,回传参数为 dog;但是dog没有age属性,fn1调用时,就不安全。type ChildToChild = (blackDog: BlackDog) => BlackDog;const fn1:ChildToChild = (blackDog) => blackDog;exec(fn1);// 参数是父类,返回值是父类 (报错)// 返回值不安全,比如示例2,需要有name属性,但是fn2的返回值没有name属性type ParentToParent = (animal: Animal) => Animal;const fn2:ParentToParent = (animal) => animal;exec(fn2);// 参数是子类,返回值是父类 (报错)// 参数、返回值都不安全type ChildToParent = (blackDog: BlackDog) => Animal;const fn3:ChildToParent = (blackDog) => animal;exec(fn3);// 参数是父类,返回值是子类(正确)type ParentToChild = (animal: Animal) => BlackDog;const fn4:ParentToChild = (animal) => blackDog;exec(fn4);
示例2:
// string|number|boolean 是 string|number 的父类型// string 是 string|number 的子类型type Cb = (a:string|number) => string|number;function exec(cb:Cb):void{}//报错type ChildToChild = (a:string) => string;let fn1:ChildToChild = (a) => 'jack';exec(fn1);// 报错type ChildToParent = (a:string) => string|number|boolean;let fn2!:ChildToParent;exec(fn2);//报错type ParentToParent = (a:string|number|boolean) => string|number|boolean;let fn3!:ParentToParent;exec(fn3);// oktype ParentToChild = (a:string|number|boolean) => string;let fn4!:ParentToChild;exec(fn4);
泛型的兼容性
泛型在判断兼容性的时候会先判断具体的类型,然后再进行兼容性判断
// 接口内容为空,没用到泛型的时候是可以的。interface Empty<T>{}let x!:Empty<string>;let y!:Empty<number>;x = y;// 接口内容不为空的时候不可以interface NotEmpty<T>{data: T,}let x2!:NotEmpty<string>;let y2!:NotEmpty<number>;x2 = y2; //报错//实现原理如下,称判断具体的类型再判断兼容性interface NotEmptyString{data:string}interface NotEmptyNumber{data:number}let xx2!:NotEmptyString;let yy2!:NotEmptyNumber;xx2 = yy2;
枚举的兼容性
- 枚举类型与数字类型兼容,并且数字类型与枚举类型兼容
- 不同枚举类型之间是不兼容的 ```typescript // 数字可以赋给枚举 enum Colors { red, green }; let c:Colors; c = Colors.red; c = 1; //ok c = ‘1’; //报错
// 枚举值可以赋给数字 let n:number; n = 1; n = Colors.red; //ok ```
