Angular單元測(cè)試系列-如何使用Jasmine進(jìn)行Angular單元測(cè)試

以下是我假定那些極少或壓根沒(méi)寫(xiě)單元測(cè)試的人準(zhǔn)備的狈谊,因此登刺,會(huì)白話解釋諸多概念性問(wèn)題,同時(shí)會(huì)結(jié)合 Jasmine 與之對(duì)應(yīng)的方法進(jìn)行講解爱葵。

一施戴、概念

Test Suite

測(cè)試套件,哪怕一個(gè)簡(jiǎn)單的類钧惧,也會(huì)有若干的測(cè)試用例暇韧,因此將這些測(cè)試用例集合在一個(gè)分類下就叫Test Suite勾习。

而在 Jasmine 就是使用 describe 全局函數(shù)來(lái)表示浓瞪,它的第一個(gè)字符串參數(shù)用來(lái)表示Suite的名稱或標(biāo)題,第二個(gè)方法參數(shù)就是實(shí)現(xiàn)Suite代碼了巧婶。

describe('test suite name', () => {
});

Specs

一個(gè)Specs相當(dāng)于一個(gè)測(cè)試用例乾颁,也就是我們實(shí)現(xiàn)測(cè)試具體代碼體。

Jasmine 就是使用 it 全局函數(shù)來(lái)表示艺栈,和 describe 類似英岭,字符串和方法兩個(gè)參數(shù)。

而每個(gè) Spec 內(nèi)包括多個(gè) expectation 來(lái)測(cè)試需要測(cè)試的代碼湿右,只要任何一個(gè) expectation 結(jié)果為 false 就表示該測(cè)試用例為失敗狀態(tài)诅妹。

describe('demo test', () => {
    const VALUE = true;
    it('should be true', () => {
        expect(VALUE).toBe(VALUE);
    })
});

Expectations

斷言,使用 expect 全局函數(shù)來(lái)表示,只接收一個(gè)代表要測(cè)試的實(shí)際值吭狡,并且需要與 Matcher 代表期望值尖殃。

二、常用方法

Matchers

斷言匹配操作划煮,在實(shí)際值與期望值之間進(jìn)行比較送丰,并將結(jié)果通知Jasmine,最終Jasmine會(huì)判斷此 Spec 成功還是失敗弛秋。

Jasmine 提供非常豐富的API器躏,一些常用的Matchers:

  • toBe() 等同 ===
  • toNotBe() 等同 !==
  • toBeDefined() 等同 !== undefined
  • toBeUndefined() 等同 === undefined
  • toBeNull() 等同 === null
  • toBeTruthy() 等同 !!obj
  • toBeFalsy() 等同 !obj
  • toBeLessThan() 等同 <
  • toBeGreaterThan() 等同 >
  • toEqual() 相當(dāng)于 ==
  • toNotEqual() 相當(dāng)于 !=
  • toContain() 相當(dāng)于 indexOf
  • toBeCloseTo() 數(shù)值比較時(shí)定義精度,先四舍五入后再比較蟹略。
  • toHaveBeenCalled() 檢查function是否被調(diào)用過(guò)
  • toHaveBeenCalledWith() 檢查傳入?yún)?shù)是否被作為參數(shù)調(diào)用過(guò)
  • toMatch() 等同 new RegExp().test()
  • toNotMatch() 等同 !new RegExp().test()
  • toThrow() 檢查function是否會(huì)拋出一個(gè)錯(cuò)誤

而這些API之前用 not 來(lái)表示負(fù)值的判斷登失。

expect(true).not.toBe(false);

這些Matchers幾乎可以滿足我們?nèi)粘P枨螅?dāng)然你也可以定制自己的Matcher來(lái)實(shí)現(xiàn)特殊需求科乎。

Setup 與 Teardown

一份干將的測(cè)試代碼很重要壁畸,因此我們可以將這些重復(fù)的 setup 與 teardown 代碼,放在與之相對(duì)應(yīng)的 beforeEachafterEach 全局函數(shù)里面茅茂。

beforeEach 表示每個(gè) Spec 執(zhí)行之前捏萍,反之。

describe('demo test', () => {
    let val: number = 0;
    beforeEach(() => {
        val = 1;
    });
    it('should be true', () => {
        expect(val).toBe(1);
    });
    it('should be false', () => {
        expect(val).not.toBe(0);
    });
});

數(shù)據(jù)共享

如同上面示例中空闲,我們可以在每個(gè)測(cè)試文件開(kāi)頭令杈、describe 來(lái)定義相應(yīng)的變量,這樣每個(gè) it 內(nèi)部可以共享它們碴倾。

當(dāng)然逗噩,每個(gè) Spec 的執(zhí)行周期間也會(huì)伴隨著一個(gè)空的 this 對(duì)象,直至 Spec 執(zhí)行結(jié)束后被清空跌榔,利用 this 也可以做數(shù)據(jù)共享异雁。

嵌套代碼

有時(shí)候當(dāng)我們對(duì)某個(gè)組件進(jìn)行測(cè)試時(shí),而這個(gè)組件會(huì)有不同狀態(tài)來(lái)展示不同的結(jié)果僧须,這個(gè)時(shí)候如果只用一個(gè) describe 會(huì)顯得不過(guò)優(yōu)雅纲刀。

因此,嵌套 describe担平,會(huì)讓測(cè)試代碼示绊、測(cè)試報(bào)告看起來(lái)更漂亮。

describe('AppComponent', () => {
    describe('Show User', () => {
        it('should be show panel.', () => {});
        it('should be show avatar.', () => {});
    });
    describe('Hidden User', () => { 
        it('should be hidden panel.', () => {});
    });
});

跳過(guò)測(cè)試代碼塊

需求總是三心二意的暂论,但好不容易寫(xiě)好的測(cè)試代碼面褐,難道要?jiǎng)h除嗎?非也……

Suites 和 Specs 分別可以用 xdescribexit 全局函數(shù)來(lái)跳過(guò)這些測(cè)試代碼塊取胎。

三展哭、配合Angular工具集

Spy

Angular的自定義事件實(shí)在太普遍了,但為了測(cè)試這些自定義事件,因此監(jiān)控事件是否正常被調(diào)用是非常重要匪傍。好在坝咐,Spy 可以用于監(jiān)測(cè)函數(shù)是否被調(diào)用,這簡(jiǎn)直就是我們的好伙伴析恢。

以下示例暫時(shí)無(wú)須理會(huì)墨坚,暫且體驗(yàn)一下:

describe('AppComponent', () => {
    let fixture: ComponentFixture<TestComponent>;
    let context: TestComponent;

    beforeEach(() => {
        TestBed.configureTestingModule({
            declarations: [TestComponent]
        });
        fixture = TestBed.createComponent(TestComponent);
        context = fixture.componentInstance;
        // 監(jiān)聽(tīng)onSelected方法
        spyOn(context, 'onSelected');
        fixture.detectChanges();
    });

    it('should be called [selected] event.', () => {
        // 觸發(fā)selected操作

        // 斷言是否被調(diào)用過(guò)
        expect(context.onSelected).toHaveBeenCalled();
    });
});

異步支持

首先,這里的異步是指帶有 Observable 或 Promise 的異步行為映挂,因此對(duì)于組件在調(diào)用某個(gè) Service 來(lái)異步獲取數(shù)據(jù)時(shí)的測(cè)試狀態(tài)泽篮。

假設(shè)我們的待測(cè)試組件代碼:

export class AppComponent {
  constructor(private _user: UserService) {}

  query() {
    this._user.quer().subscribe(() => {});
  }
}

async

async 無(wú)任何參數(shù)與返回值,所有包裹代碼塊里的測(cè)試代碼柑船,可以通過(guò)調(diào)用 whenStable()所有待處理異步行為都完成后再進(jìn)行回調(diào)帽撑;最后,再進(jìn)行斷言操作鞍时。

it('should be get user list (async)', async(() => {
    // call component.query();
    fixture.whenStable().then(() => {
        fixture.detectChanges();
        expect(true).toBe(true);
    });
}));

fakeAsync

如果說(shuō) async 還需要回調(diào)才能進(jìn)行斷點(diǎn)讓你受不了的話亏拉,那么 fakeAsync 可以解決這一點(diǎn)。

it('should be get user list (async)', fakeAsync(() => {
    // call component.query();
    tick();
    fixture.detectChanges();
    expect(true).toBe(true);
}));

這里只是將回調(diào)換成 tick()逆巍,怎么樣及塘,是不是很酷。

Jasmine自帶異步

如前面所說(shuō)的異步是指帶有 Observable 或 Promise 的異步行為锐极,而有時(shí)候我們有些東西是依賴 setTimeout 或者可能是需要外部訂閱結(jié)果以后才能觸發(fā)時(shí)怎么辦呢笙僚?

可以使用 done() 方法。

it('async demo', (done: () => void) => {
    context.show().subscribe(res => {
        expect(true).toBe(true);
        done();
    });
    el.querySelected('xxx').click();
});

四灵再、結(jié)論

本章幾乎所有的內(nèi)容在Angular單元測(cè)試經(jīng)常使用到的東西肋层;特別是異步部分,三種不同異步方式并非共存的翎迁,而是需要根據(jù)具體業(yè)務(wù)而采用栋猖。否則,你會(huì)發(fā)現(xiàn)真TM難寫(xiě)單元測(cè)試汪榔。畢竟這是一個(gè)異步的世界蒲拉。

自此,我們算是為Angular寫(xiě)單元測(cè)試打下了基礎(chǔ)揍异。后續(xù)全陨,將不會(huì)再對(duì)這類基礎(chǔ)進(jìn)行解釋爆班。

那么下一篇衷掷,我們將介紹Component、Directive柿菩、Pipe 以及Service單元測(cè)試戚嗅。

happy coding!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子懦胞,更是在濱河造成了極大的恐慌替久,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,265評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件躏尉,死亡現(xiàn)場(chǎng)離奇詭異蚯根,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)胀糜,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門颅拦,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人教藻,你說(shuō)我怎么就攤上這事距帅。” “怎么了括堤?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,852評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵碌秸,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我悄窃,道長(zhǎng)讥电,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,408評(píng)論 1 283
  • 正文 為了忘掉前任轧抗,我火速辦了婚禮允趟,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘鸦致。我一直安慰自己潮剪,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,445評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布分唾。 她就那樣靜靜地躺著抗碰,像睡著了一般。 火紅的嫁衣襯著肌膚如雪绽乔。 梳的紋絲不亂的頭發(fā)上弧蝇,一...
    開(kāi)封第一講書(shū)人閱讀 49,772評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音折砸,去河邊找鬼看疗。 笑死,一個(gè)胖子當(dāng)著我的面吹牛睦授,可吹牛的內(nèi)容都是我干的两芳。 我是一名探鬼主播,決...
    沈念sama閱讀 38,921評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼去枷,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼怖辆!你這毒婦竟也來(lái)了是复?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,688評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤竖螃,失蹤者是張志新(化名)和其女友劉穎淑廊,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體特咆,經(jīng)...
    沈念sama閱讀 44,130評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡季惩,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,467評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了腻格。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片蜀备。...
    茶點(diǎn)故事閱讀 38,617評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖荒叶,靈堂內(nèi)的尸體忽然破棺而出碾阁,到底是詐尸還是另有隱情,我是刑警寧澤些楣,帶...
    沈念sama閱讀 34,276評(píng)論 4 329
  • 正文 年R本政府宣布脂凶,位于F島的核電站,受9級(jí)特大地震影響愁茁,放射性物質(zhì)發(fā)生泄漏蚕钦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,882評(píng)論 3 312
  • 文/蒙蒙 一鹅很、第九天 我趴在偏房一處隱蔽的房頂上張望嘶居。 院中可真熱鬧,春花似錦促煮、人聲如沸邮屁。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,740評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)佑吝。三九已至,卻和暖如春绳匀,著一層夾襖步出監(jiān)牢的瞬間芋忿,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,967評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工疾棵, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留戈钢,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,315評(píng)論 2 360
  • 正文 我出身青樓是尔,卻偏偏與公主長(zhǎng)得像殉了,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子嗜历,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,486評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容