Typescript中的协变、逆变、双向协变

协变(Covariant)、逆变(Contravariant)、双向协变(Bivariant)并非Typescript所特有,其他结构化语言诸如c#、java等也都拥有该特性。
怎么理解这个概念呢? 先说说集合、超集、子集(set, superset, subset)

下图中有两个集合:脊索动物、哺乳动物。 哺乳动物一定是脊索动物,反之则不一定。 因此我们说脊索动物是哺乳动物的超集,哺乳动物是脊索动物的子集。

哺乳动物一定具有脊索动物的特性,反之则不一定。

协变
协变是指:子集能赋值给其超集。

class Chordate {
    hasSpine(): boolean {
        return true;
    }
}

class Mammal extends Chordate {
    canBreastFeed(): boolean {
        return true;
    }
}

function foo(animal: Chordate){
    animal.hasSpine();
}

foo(new Chordate());
foo(new Mammal());

以上代码证明了Typescript支持协变,Mammal是Chordate的子集,方法foo接受参数类型为Chordate,而Mammal实例也能赋值给Chordate参数。

逆变
逆变(Contravariance)与双变(Bivariance)只针对函数有效。 --strictFunctionTypes 开启时只支持逆变,关闭时支持双变。

class Chordate {
    hasSpine(): boolean {
        return true;
    }
}

class Mammal extends Chordate {
    canBreastFeed(): boolean {
        return true;
    }
}

declare let f1: (x: Chordate) => void;
declare let f2: (x: Mammal) => void;

f2=f1;
f1=f2; //Error: Mammal is incompatible with Chordate

协变比较好理解,为什么函数赋值,只能支持逆变(默认),而不支持协变呢? 请看以下代码示例

class Animal {
    doAnimalThing(): void {
        console.log("I am a Animal!")
    }
}

class Dog extends Animal {
    doDogThing(): void {
        console.log("I am a Dog!")
    }
}

class Cat extends Animal {
    doCatThing(): void {
        console.log("I am a Cat!")
    }
}

function makeAnimalAction(animalAction: (animal: Animal) => void) : void {
    let cat: Cat = new Cat()
    animalAction(cat)
}

function dogAction(dog: Dog) {
    dog.doDogThing()
}

makeAnimalAction(dogAction) // TS Error at compilation, since we are trying to use `doDogThing()` to a `Cat`

上述代码说明了如果函数赋值支持协变的话,有可能会导致bug

参考:
https://codethrasher.com/post/2019-08-28-type-variance-and-typescript/
https://www.typescriptlang.org/docs/handbook/release-notes/typescript-2-6.html
https://dev.to/codeozz/how-i-understand-covariance-contravariance-in-typescript-2766

posted @ 2021-06-29 17:42  老胡Andy  阅读(857)  评论(0编辑  收藏  举报