国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

玩轉Node.js單元測試

開發 后端
代碼部署之前,進行一定的單元測試是十分必要的,這樣能夠有效并且持續保證代碼質量。而實踐表明,高質量的單元測試還可以幫助我們完善自己的代碼。這篇博客將通過一些簡單的測試案例,介紹幾款Node.js測試模塊: Mocha和Should,SuperTest。

代碼部署之前,進行一定的單元測試是十分必要的,這樣能夠有效并且持續保證代碼質量。而實踐表明,高質量的單元測試還可以幫助我們完善自己的代碼。這篇博客將通過一些簡單的測試案例,介紹幾款Node.js測試模塊: Mocha和Should,SuperTest。本文側重于解釋原理,各個模塊的詳細使用案例以后單獨再聊。

為啥需要單元測試?

所謂單元測試,就是對某個函數或者API進行正確性驗證。來看個簡單的例子add1.js:

function add(a, b) {
    return a + b;
}

沒錯,我寫了一個加法函數。這有啥好測的呢?不妨用node執行一下:

> add = function(a, b){return a + b}
[Function: add] > add(4) NaN

當add函數僅給定一個參數4的時候,a為4,b為undefined,兩者相加為NaN。

  • 你考慮過只有一個參數的場景嗎?
  • 給定一個參數時,NaN是你想要的結果嗎?
  • 如果參數不是整數怎么辦?

這時,就需要單元測試來驗證各種可能的場景了。

如果我把add函數定義為兩個整數相加,而其他輸入則返回undefined,那么正確的代碼add2.js應該是這樣的:

function add(a, b) {
    if (typeof a === "number" && typeof b === "number")
    {
        return a + b;
    }
    else
    {
        return undefined;
    }
}

發現一個有趣的現象,我們寫代碼的時候很容易陷入思維漏洞,而寫測試的時候往往會考慮各種情況,這就是所謂的TDD(Test-Driven-Development: 測試驅動開發)的神奇之處。因此,進行一定的單元測試是十分必要的:

  • 驗證代碼的正確性
  • 避免修改代碼時出錯
  • 避免其他團隊成員修改代碼時出錯
  • 便于自動化測試與部署

測試框架 - Mocha

下面的測試代碼test2.js用于測試add2.js。這里使用了測試框架Mocha以及Node.js自帶的斷言庫Assert。

var add = require("../add2.js");
var assert  = require("assert");
// 當2個參數均為整數時
it("should return 3", function() {
    var sum = add(1, 2);
    assert.equal(sum, 3);
});
// 當第2個參數為String時
it("should return undefined", function() {
    var sum = add(1, "2");
    assert.equal(sum, undefined);
});
// 當只有1個參數時
it("should return undefined", function() {
    var sum = add(1);
    assert.equal(sum, undefined);
});

測試代碼中使用了測試框架Mocha提供的it函數,3個it函數分別測試了3種不同的案例(test case)。it函數的第1個參數為字符串,用于描述測試,一般會寫期望得到的結果,例如”should return 3”; 而第2個參數為函數,用于編寫測試代碼,一般是先調用被測試的函數或者API,獲取結果之后,使用斷言庫判斷執行結果是否正確。

測試代碼中使用了Node.js自帶的斷言庫Assert的assert.equal函數,用于判定add函數返回的結果是否正確。assert.equal成功時不會發生什么,而失敗時會拋出一個AssertionError。不妨使用node測試一下:

> assert  = require("assert");
> assert.equal(1, 1);
undefined
> assert.equal(1, 2);
AssertionError: 1 == 2
    at repl:1:8
    at sigintHandlersWrap (vm.js:22:35)
    at sigintHandlersWrap (vm.js:96:12)
    at ContextifyScript.Script.runInThisContext (vm.js:21:12)
    at REPLServer.defaultEval (repl.js:313:29)
    at bound (domain.js:280:14)
    at REPLServer.runBound [as eval] (domain.js:293:12)
    at REPLServer.<anonymous> (repl.js:513:10)
    at emitOne (events.js:101:20)
    at REPLServer.emit (events.js:188:7)

原理:

我們按照Mocha的it函數編寫一個個測試案例,然后Mocha負責執行這些案例;當assert.equal斷言成功時,則測試案例通過;當assert.equal斷言失敗時,拋出AssertionError,Mocha能夠捕獲到這些異常,然后對應的測試案例失敗。

使用mocha執行test2.js:

mocha test/test2.js

下面為輸出,表示測試案例全部通過

✓ should return 3
✓ should return undefined
✓ should return undefined
3 passing

而當我們使用test1.js測試add1.js時,則后面2個測試案例失敗:

✓ should return 3
  1) should return undefined
  2) should return undefined
  1 passing (14ms)
  2 failing
  1)  should return undefined:
     AssertionError: '12' == undefined
      at Context.<anonymous> (test/test1.js:18:12)
  2)  should return undefined:
     AssertionError: NaN == undefined
      at Context.<anonymous> (test/test1.js:25:12)

斷言庫 - Should

Node.js自帶的斷言庫Assert提供的函數有限,在實際工作中,Should等第三方斷言庫則更加強大和實用。

我寫了一個merge函數merge.js,實現了類似于_.extend()Object.assign()的功能,用于合并兩個Object的屬性。

function merge(a, b) {
    if (typeof a === "object" && typeof b === "object")
    {
        for (var property in b)
        {
            a[property] = b[property];
        }
        return a;
    }
    else
    {
        return undefined;
    }
}

然后我使用Should寫了對應的測試代碼test3.js:

require("should");
var merge = require("../merge.js");
// 當2個參數均為對象時
it("should success", function() {
    var a = {
        name: "Fundebug",
        type: "SaaS"
    };
    var b = {
        service: "Real time bug monitoring",
        product:
        {
            frontend: "JavaScript",
            backend: "Node.js",
            mobile: "微信小程序"
        }
    };
    var c = merge(a, b);
    c.should.have.property("name", "Fundebug");
    c.should.have.propertyByPath("product", "frontend").equal("JavaScript");
});
// 當只有1個參數時
it("should return undefined", function() {
    var a = {
        name: "Fundebug",
        type: "SaaS"
    };
    var c = merge(a);
    (typeof c).should.equal("undefined");
});

測試代碼稍微有點長,但是使用Should的只有三處:

c.should.have.property("name", "Fundebug");
c.should.have.propertyByPath("product", "frontend").equal("JavaScript");
(typeof c).should.equal("undefined");

可知Should能夠:

  • 驗證對象是否存在某屬性,并驗證其取值
  • 驗證對象是否存在某個嵌套屬性,并使用鏈式方式驗證其取值

那么Should為什么不能直接驗證c的取值為undefined呢?比如這樣寫:

c.should.equal(undefined); // 這樣寫是錯誤的

原理:

Should會為每個對象添加should屬性,然后通過該屬性提供各種斷言函數,我們可以使用這些函數驗證對象的取值。對于undefined,Should無法為其添加屬性,因此失敗。

通過node驗證發現,導入Should之后,空對象a增加了一個should屬性。

> a = {}
> typeof a.should
'undefined'
> require("should")
> typeof a.should
'object'

測試HTTP接口 - SuperTest

Node.js是用于后端開發的語言,而后端開發其實很大程度上等價于編寫HTTP接口,為前端提供服務。那么,Node.js單元測試則少不了對HTTP接口進行測試。

我用Node.js自帶的HTTP模塊寫了一個簡單的HTTP接口server.js

var http = require("http");
var server = http.createServer((req, res) =>
{
    res.writeHead(200,
    {
        "Content-Type": "text/plain"
    });
    res.end("Hello Fundebug");
});
server.listen(8000);

按照Mocha的原理,測試HTTP接口并不難: 訪問接口; 獲取返回數據; 驗證返回結果。使用Node.js原生的http與assert模塊就可以了test4.js:

require("../server.js");
var http = require("http");
var assert = require("assert");
it("should return hello fundebug", function(done) {
    http.get("http://localhost:8000", function(res)     {
        res.setEncoding("utf8");
        res.on("data", function(text)         {
            assert.equal(res.statusCode, 200);
            assert.equal(text, "Hello Fundebug");
            done();
        });
    });
});

值得稍微注意的一點是,http.get訪問HTTP接口是一個異步操作。Mocha在測試異步代碼是需要為it函數添加回調函數done,在斷言結束的地方調用done,這樣Mocha才能知道什么時候結束這個測試。

既然Node.js自帶的模塊就能夠測試HTTP接口了,為什么還需要SuperTest呢?不妨先看一下測試代碼test5.js:

var request = require("supertest");
var server = require("../server.js");
var assert = require("assert");
it("should return hello fundebug", function(done) {
    request(server)
        .get("/")
        .expect(200)
        .expect(function(res)         {
            assert.equal(res.text, "Hello Fundebug");
        })
        .end(done);
});

對比兩個測試代碼,會發現后者簡潔很多。

原理

SuperTest封裝了發送HTTP請求的接口,并且提供了簡單的expect斷言來判定接口返回結果。對于POST接口,使用SuperTest的優勢將更加明顯,因為使用Node.js的http模塊發送POST請求是很麻煩的。

要做多少單元測試?

本文所寫的單元測試案例,都很簡單。然而,在實際工作中,單元測試是一個很頭痛的事情。修改了代碼有時意味著必須修改單元測試,寫了新的函數或者API就得寫新的單元測試。如果較真起來,單元測試可以沒完沒了地寫,但這是沒有意義的。而根據二八原理,20%的測試可以解決80%的問題。剩下的20%問題,事實上我們是力不從心的。換句話說,想通過測試消除所有BUG,是不現實的。

因此,對生產代碼進行實時錯誤監測是非常有必要的,這也是我們Fundebug努力在做的事情。

參考鏈接

責任編輯:張燕妮 來源: KiwenLau
相關推薦

2012-10-24 14:56:30

IBMdw

2017-01-14 23:42:49

單元測試框架軟件測試

2013-11-01 09:34:56

Node.js技術

2015-03-10 10:59:18

Node.js開發指南基礎介紹

2017-01-14 23:26:17

單元測試JUnit測試

2017-01-16 12:12:29

單元測試JUnit

2020-08-18 08:10:02

單元測試Java

2023-07-26 08:58:45

Golang單元測試

2011-07-04 18:16:42

單元測試

2020-05-07 17:30:49

開發iOS技術

2021-05-05 11:38:40

TestNGPowerMock單元測試

2017-03-23 16:02:10

Mock技術單元測試

2011-11-01 10:30:36

Node.js

2011-09-02 14:47:48

Node

2011-09-09 14:23:13

Node.js

2011-09-08 13:46:14

node.js

2011-11-10 08:55:00

Node.js

2011-05-16 16:52:09

單元測試徹底測試

2024-04-26 09:44:39

2021-12-25 22:29:57

Node.js 微任務處理事件循環
點贊
收藏

51CTO技術棧公眾號

丝袜美腿高跟呻吟高潮一区| 一本一道久久a久久精品蜜桃 | 欧美在线视频免费观看| 美女av在线播放| 午夜不卡av免费| 黄色直播在线| 亚洲欧美日韩在线播放| 国产精品涩涩涩视频网站| 国产精品中文欧美| 真人做人试看60分钟免费| 日产国产高清一区二区三区| 日本不卡二区高清三区| 久久精品伊人| 性欧美videosex高清少妇| 午夜一级久久| 日韩久久在线| 国产精品一二三四五| 国产免费xxx| 寂寞少妇一区二区三区| 久久精品在线免费视频| 国产原创一区二区| 免费毛片网站在线观看| 97国产精品videossex| 国产日产欧美视频| 国产精品久久影院| 午夜黄色一级片| 欧美午夜激情在线| 乱人伦中文视频在线| 精品久久久久久久人人人人传媒 | 欧美精品xxxxbbbb| 国产在线高清理伦片a| 欧美不卡一区二区三区| 国产精品一区二区av影院萌芽| 国产一区二区精品丝袜| 国产亚洲亚洲国产一二区| 国模精品视频一区二区| 波多野结衣在线播放一区| 91免费看网站| 精品在线亚洲视频| 欧美国产日韩在线播放| 亚洲成人午夜影院| av软件在线观看| 色婷婷久久av| 日韩av大片| 精品国产一区二区三区麻豆小说| 麻豆久久久久久| 久久精品国产精品亚洲色婷婷| **欧美大码日韩| 成在在线免费视频| 亚洲另类激情图| 亚洲码欧美码一区二区三区| 国产日韩中文字幕| 天堂va蜜桃一区二区三区漫画版| 18禁网站免费无遮挡无码中文| 亚洲综合色丁香婷婷六月图片| 粗大黑人巨茎大战欧美成人| 日韩视频亚洲视频| 日韩精品水蜜桃| 亚洲欧洲一区二区福利| 国产精品久久久一区麻豆最新章节| 嫩草精品影院| 深夜福利亚洲导航| 欧美a级在线| 国产精品后入内射日本在线观看| 亚洲香蕉伊在人在线观| av福利导福航大全在线| 欧美孕妇孕交黑巨大网站| 亚洲综合99| jizzjizzjizz亚洲女| 欧美一级片在线观看| 久久夜色电影| 色噜噜狠狠色综合网| 亚洲欧洲制服丝袜| 悠悠资源网亚洲青| 成人黄色免费片| 99久久99久久精品免费看蜜桃| 国产在线一二| 久久久久久久久久久免费精品| 石原莉奈一区二区三区在线观看| 成人黄网大全在线观看| 精品免费99久久| 91视频一区| 日韩手机在线观看视频| 欧美一区二区三区日韩| 妖精视频一区二区三区| 999一区二区三区| 91搞黄在线观看| 日本欧美韩国国产| www.xxx麻豆| 日韩一区二区三区观看| 久操精品在线| 男人亚洲天堂网| 精品欧美黑人一区二区三区| 91视频久久| 天天干天天操天天做| 日韩经典中文字幕| 日韩一级大片| 视频一区二区三区国产 | 日韩av超清在线观看| 国产欧美亚洲日本| 亚洲色图制服诱惑| 国产一区一一区高清不卡| 久久资源亚洲| 欧美亚洲综合色| 日韩在线观看一区 | 91精品久久久久久久久中文字幕| 99久久综合国产精品| 女子免费在线观看视频www| 国产精品久久综合av爱欲tv| 久久亚洲欧美国产精品乐播| 国产精品专区免费| 欧美一级二级三级| 欧美日韩免费一区二区三区视频| 国产不卡一区| www.夜夜爽| 欧美激情一二三| av成人免费在线观看| 一区二区三区四区日本视频| 日本一区二区三区四区高清视频| 欧美日韩在线三级| 黄色日韩在线| www黄在线观看| 狠狠色综合一区二区| 在线精品视频小说1| 999久久久免费精品国产| 嫩草影院网站在线| 国产精品av电影| 亚洲激情男女视频| 日韩三级视频| 欧美高清xxxxxkkkkk| 国产精品视频一区国模私拍 | 亚洲成人av在线播放| 国产精品毛片| 国产色婷婷在线| 亚洲欧美日韩综合一区| 亚洲国产中文字幕久久网| 日韩av中文字幕一区二区| 中文字幕在线观看网站| 日韩视频在线观看国产| 日韩久久精品电影| www.亚洲色图| 久久久久久亚洲精品美女| 777久久久精品一区二区三区 | 国产精品免费一区二区| 色婷婷亚洲综合| 影音先锋在线一区| 最爽无遮挡行房视频在线| 西游记1978| 最近更新的2019中文字幕| 99国内精品久久| 99re8这里有精品热视频8在线| 向日葵污视频在线观看| 欧洲成人性视频| 欧美特黄级在线| 西西人体一区二区| 欧美成人影院| 日韩视频免费在线播放| 国产成人91久久精品| 在线观看免费一区| 毛片基地黄久久久久久天堂| 激情久久99| 午夜3点看的视频| 国产一级二级三级精品| 亚洲第一福利网| 久久先锋资源网| 日韩一区三区| 丰乳肥臀在线| 免费在线激情视频| 国产男女猛烈无遮挡91| 精品不卡在线视频| 国产欧美日韩在线视频| 欧美伊人影院| 在线一区av| av天在线播放| 欧美精品123| 久久久成人精品| 欧美性xxxx极品高清hd直播| 国产综合色精品一区二区三区| 免费福利视频一区| 在线视频观看国产| 亚欧激情乱码久久久久久久久| 国产另类自拍| 欧美乱大交xxxxx另类电影| 色婷婷一区二区三区四区| 粉嫩aⅴ一区二区三区四区| 国产成人av| 国产日韩电影| 免费一级毛片在线观看| 欧美大片在线播放| 超碰97在线资源| 久久久久99精品久久久久| 日韩欧美一区二区在线| 成人va在线观看| 欧美aa国产视频| 精品中文在线| 美女日批视频在线观看| 猛男欧美办公室激情在线| 欧美亚洲黄色片| 精品亚洲欧美日韩| 欧美中文字幕精品|