typescript中聲明合并介紹
聲明合并
類型合并表明編譯器將合并兩個分開的并且名稱相同的聲明,合并之后的聲明擁有兩個聲明的特點,任意數(shù)量的聲明可以被合并,不僅限兩個。
合并Interface
1.interface的非函數(shù)成員應該是唯一的,如果兩個interface都聲明一個名稱相同但類型不同的非函數(shù)成員,編譯器將提示錯誤:
interface Box {
height: number;
}
interface Box {
height: string;
}
2.對于函數(shù)成員,每個相同名稱的成員被看作是相同名稱函數(shù)的重載,但是當出現(xiàn)兩個interface時,第二個有更高的優(yōu)先級,會覆蓋前一個:
interface Cloner {
clone(animal: Animal): Animal;
}
interface Cloner {
clone(animal: Sheep): Sheep;
}
interface Cloner {
clone(animal: Dog): Dog;
clone(animal: Cat): Cat;
}
// 最終的排序是
interface Cloner {
clone(animal: Dog): Dog;
clone(animal: Cat): Cat;
clone(animal: Sheep): Sheep;
clone(animal: Animal): Animal;
}當然這個規(guī)則有一個例外,當函數(shù)的參數(shù)類型是一個單字面量類型(single string literal type),它將會根據(jù)優(yōu)先級排序,并放在聲明頂部:
interface Document {
createElement(tagName: any): Element;
}
interface Document {
createElement(tagName: 'div'): HTMLDivElement;
createElement(tagName: 'span'): HTMLSpanElement;
}
interface Document {
createElement(tagName: string): HTMLElement;
createElement(tagName: 'canvas'): HTMLCanvasElement;
}
// 字面量根據(jù)冒泡排序并放在了聲明頂部
interface Document {
createElement(tagName: 'canvas'): HTMLCanvasElement;
createElement(tagName: 'div'): HTMLDivElement;
createElement(tagName: 'span'): HTMLSpanElement;
createElement(tagName: string): HTMLElement;
createElement(tagName: any): Element;
}合并Namespace
- 合并兩個相同名稱的
namespace時,將進一步添加第二個namespace中導出的成員到第一個namespace。
namespace Animals {
export class Zebra {}
}
namespace Animals {
export interface Legged {
numberOfLegs: number;
}
export class Dog {}
}
// 合并到了第一個
namespace Animals {
export interface Legged {
numberOfLegs: number;
}
export class Zebra {}
export class Dog {}
}- 當一個
namespace發(fā)生合并時,和它合并的namesapce不能訪問它的未導出的成員:
namespace Animal {
const haveMuscles = true;
export function animalsHaveMuscles() {
return haveMuscles;
}
}
namespace Animal {
export function doAnimalsHaveMuscles() {
return haveMuscles; // Error, because haveMuscles is not accessible here
}
}
可以看到無法訪問haveMuscles,同時運行也會報錯,可以結合編譯后的例子看:

namespace和class、enum、function合并
- 和合并
namespace一樣,class可以訪問namespace中導出的類型和值:
class Album {
label: Album.AlbumLabel;
}
namespace Album {
export class AlbumLabel {}
}namespace和function合并可以像javascript那樣在方法上添加屬性:
function buildLabel(name: string): string {
return buildLabel.prefix + name + buildLabel.suffix;
}
namespace buildLabel {
export const suffix = '';
export const prefix = 'Hello, ';
}
console.log(buildLabel('Sam Smith'));可以看編譯之后的代碼,可以看到直接在buildLabel上添加了屬性:

namespace和enum發(fā)生合并時,namespace可以擴展enum
enum Color {
red = 1,
green = 2,
blue = 4,
}
namespace Color {
export function mixColor(colorName: string) {
if (colorName == 'yellow') {
return Color.red + Color.green;
} else if (colorName == 'white') {
return Color.red + Color.green + Color.blue;
} else if (colorName == 'magenta') {
return Color.red + Color.blue;
} else if (colorName == 'cyan') {
return Color.green + Color.blue;
}
}
}可以看編譯之后的:

class之間不允許合并,但是如果需要模仿類似功能,可以參照 Mixins in Typscripts
Module擴展
盡管Module之間是不支持合并的,但是你可以通過導入需要擴展的方法,然后再更改它,這種方式去實現(xiàn):
// observable.ts
export class Observable<T> {
// ... implementation left as an exercise for the reader ...
}
// map.ts
import { Observable } from "./observable";
Observable.prototype.map = function (f) {
// ... another exercise for the reader
};但是這樣編譯器并不能提供良好的提示,所以需要擴展module的聲明:
// observable.ts
export class Observable<T> {
// ... implementation left as an exercise for the reader ...
}
// map.ts
import { Observable } from "./observable";
declare module "./observable" {
interface Observable<T> {
map<U>(f: (x: T) => U): Observable<U>;
}
}
// 擴展聲明
Observable.prototype.map = function (f) {
// ... another exercise for the reader
};
// consumer.ts
import { Observable } from "./observable";
import "./map";
let o: Observable<number>;
o.map((x) => x.toFixed());全局擴展
如果在模塊中,也可以在全局聲明中來擴展:
// observable.ts
export class Observable<T> {
// ... still no implementation ...
}
// 在這里擴展
declare global {
interface Array<T> {
toObservable(): Observable<T>;
}
}
Array.prototype.toObservable = function () {
// ...
};到此這篇關于typescript中聲明合并介紹的文章就介紹到這了,更多相關typescript聲明合并內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

