摘要:四結(jié)論本章幾乎所有的內(nèi)容在單元測試經(jīng)常使用到的東西特別是異步部分,三種不同異步方式并非共存的,而是需要根據(jù)具體業(yè)務(wù)而采用。否則,你會發(fā)現(xiàn)真難寫單元測試。自此,我們算是為寫單元測試打下了基礎(chǔ)。
以下是我假定那些極少或壓根沒寫單元測試的人準(zhǔn)備的,因此,會白話解釋諸多概念性問題,同時(shí)會結(jié)合 Jasmine 與之對應(yīng)的方法進(jìn)行講解。
一、概念 Test Suite測試套件,哪怕一個(gè)簡單的類,也會有若干的測試用例,因此將這些測試用例集合在一個(gè)分類下就叫Test Suite。
而在 Jasmine 就是使用 describe 全局函數(shù)來表示,它的第一個(gè)字符串參數(shù)用來表示Suite的名稱或標(biāo)題,第二個(gè)方法參數(shù)就是實(shí)現(xiàn)Suite代碼了。
describe("test suite name", () => { });Specs
一個(gè)Specs相當(dāng)于一個(gè)測試用例,也就是我們實(shí)現(xiàn)測試具體代碼體。
Jasmine 就是使用 it 全局函數(shù)來表示,和 describe 類似,字符串和方法兩個(gè)參數(shù)。
而每個(gè) Spec 內(nèi)包括多個(gè) expectation 來測試需要測試的代碼,只要任何一個(gè) expectation 結(jié)果為 false 就表示該測試用例為失敗狀態(tài)。
describe("demo test", () => { const VALUE = true; it("should be true", () => { expect(VALUE).toBe(VALUE); }) });Expectations
斷言,使用 expect 全局函數(shù)來表示,只接收一個(gè)代表要測試的實(shí)際值,并且需要與 Matcher 代表期望值。
二、常用方法 Matchers斷言匹配操作,在實(shí)際值與期望值之間進(jìn)行比較,并將結(jié)果通知Jasmine,最終Jasmine會判斷此 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)用過
toHaveBeenCalledWith() 檢查傳入?yún)?shù)是否被作為參數(shù)調(diào)用過
toMatch() 等同 new RegExp().test()
toNotMatch() 等同 !new RegExp().test()
toThrow() 檢查function是否會拋出一個(gè)錯(cuò)誤
而這些API之前用 not 來表示負(fù)值的判斷。
expect(true).not.toBe(false);
這些Matchers幾乎可以滿足我們?nèi)粘P枨螅?dāng)然你也可以定制自己的Matcher來實(shí)現(xiàn)特殊需求。
Setup 與 Teardown一份干將的測試代碼很重要,因此我們可以將這些重復(fù)的 setup 與 teardown 代碼,放在與之相對應(yīng)的 beforeEach 與 afterEach 全局函數(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è)測試文件開頭、describe 來定義相應(yīng)的變量,這樣每個(gè) it 內(nèi)部可以共享它們。
當(dāng)然,每個(gè) Spec 的執(zhí)行周期間也會伴隨著一個(gè)空的 this 對象,直至 Spec 執(zhí)行結(jié)束后被清空,利用 this 也可以做數(shù)據(jù)共享。
嵌套代碼有時(shí)候當(dāng)我們對某個(gè)組件進(jìn)行測試時(shí),而這個(gè)組件會有不同狀態(tài)來展示不同的結(jié)果,這個(gè)時(shí)候如果只用一個(gè) describe 會顯得不過優(yōu)雅。
因此,嵌套 describe,會讓測試代碼、測試報(bào)告看起來更漂亮。
describe("AppComponent", () => { describe("Show User", () => { it("should be show panel.", () => {}); it("should be show avatar.", () => {}); }); describe("Hidden User", () => { it("should be hidden panel.", () => {}); }); });跳過測試代碼塊
需求總是三心二意的,但好不容易寫好的測試代碼,難道要?jiǎng)h除嗎?非也……
Suites 和 Specs 分別可以用 xdescribe 和 xit 全局函數(shù)來跳過這些測試代碼塊。
三、配合Angular工具集 SpyAngular的自定義事件實(shí)在太普遍了,但為了測試這些自定義事件,因此監(jiān)控事件是否正常被調(diào)用是非常重要。好在,Spy 可以用于監(jiān)測函數(shù)是否被調(diào)用,這簡直就是我們的好伙伴。
以下示例暫時(shí)無須理會,暫且體驗(yàn)一下:
describe("AppComponent", () => { let fixture: ComponentFixture異步支持; let context: TestComponent; beforeEach(() => { TestBed.configureTestingModule({ declarations: [TestComponent] }); fixture = TestBed.createComponent(TestComponent); context = fixture.componentInstance; // 監(jiān)聽onSelected方法 spyOn(context, "onSelected"); fixture.detectChanges(); }); it("should be called [selected] event.", () => { // 觸發(fā)selected操作 // 斷言是否被調(diào)用過 expect(context.onSelected).toHaveBeenCalled(); }); });
首先,這里的異步是指帶有 Observable 或 Promise 的異步行為,因此對于組件在調(diào)用某個(gè) Service 來異步獲取數(shù)據(jù)時(shí)的測試狀態(tài)。
假設(shè)我們的待測試組件代碼:
export class AppComponent { constructor(private _user: UserService) {} query() { this._user.quer().subscribe(() => {}); } }
async
async 無任何參數(shù)與返回值,所有包裹代碼塊里的測試代碼,可以通過調(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
如果說 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自帶異步
如前面所說的異步是指帶有 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單元測試經(jīng)常使用到的東西;特別是異步部分,三種不同異步方式并非共存的,而是需要根據(jù)具體業(yè)務(wù)而采用。否則,你會發(fā)現(xiàn)真TM難寫單元測試。畢竟這是一個(gè)異步的世界。
自此,我們算是為Angular寫單元測試打下了基礎(chǔ)。后續(xù),將不會再對這類基礎(chǔ)進(jìn)行解釋。
那么下一篇,我們將介紹Component、Directive、Pipe 以及Service單元測試。
happy coding!
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/87038.html
摘要:整個(gè)系列差不多涵蓋或滿足日常單元測試開發(fā)所需的知識,當(dāng)然,像前面說的,你也認(rèn)同單元測試的重要性,否則看這系列并無任何意義因?yàn)?,我并不描述任何有關(guān)于開發(fā)技巧。 Angular單元測試在所有前端框架當(dāng)中不要太爽了,但是要全面說好它,還真需要很長的篇幅,因?yàn)楦鞣N測試方式就如同在寫一份Angular入門級開發(fā)一樣。因此,我打算使用一個(gè)系列來說明。 當(dāng)然,一切的前提是,你同我一樣認(rèn)同單元測試的重...
摘要:單元測試我們可以將其分成兩類獨(dú)立單獨(dú)測試與測試工具集。工具集還有更多,這一切我們將在單元測試組件與指令單元測試逐一說明。那么下一篇,我們將介紹如何使用進(jìn)行單元測試。 本文將探討如何搭建測試環(huán)境、以及Angular測試工具集。 測試環(huán)境 絕大部分都是利用Angular Cli來創(chuàng)建項(xiàng)目,因此,默認(rèn)已經(jīng)集成我們所需要的npm包與腳本;當(dāng)然,如果你是使用自建或官網(wǎng) quickstart 的話...
摘要:小結(jié)我們初步了解了使用來進(jìn)行測試,這樣有利于我們接下來去理解使用進(jìn)行單元測試翻譯中 showImg(https://segmentfault.com/img/bVxI9p); 這是一些列文章,陸續(xù)翻譯整理中... 原文地址:http://www.bradoncode.com/blog/2015/05/12/angularjs-testing-getting-started/ @ Bra...
摘要:原文地址上面一篇文章簡單介紹了如何使用進(jìn)行的單元測試我們用了一段簡單的代碼進(jìn)行計(jì)算的測試。添加測試接下來終于到了我們的主題,添加一些單元測試給我們忽略代碼中部分,主要集中在的代碼中。 原文地址:http://www.bradoncode.com/blog/2015/05/17/angularjs-testing-controller/@Bradley Braithwaite show...
摘要:前面我們寫過了的一篇文章開始對進(jìn)行單元測試而提供了非常有用的工具去幫助我們進(jìn)行的測試。接下來,會增加一些內(nèi)容寫測試用例接下來我們可以開始進(jìn)行測試了,我們在命令行工具里輸入下面的命令我們將下面的代碼粘貼到中去。 showImg(https://segmentfault.com/img/bVx65M); 緊隨前文如何對Angular Controller進(jìn)行單元測試,但是我們也提到了前文工...
閱讀 1275·2023-04-26 01:38
閱讀 1472·2021-11-15 11:39
閱讀 3264·2021-09-22 15:43
閱讀 2659·2019-08-30 15:55
閱讀 2059·2019-08-30 14:17
閱讀 2861·2019-08-29 14:16
閱讀 3071·2019-08-26 18:36
閱讀 2616·2019-08-26 12:19