關於單元測試的思考--Asp.Net Core單元測試最佳實踐

来源:https://www.cnblogs.com/yubaolee/archive/2018/07/07/DotNetCoreUnitTest.html
-Advertisement-
Play Games

在我們碼字過程中,單元測試是必不可少的。但在從業過程中,很多開發者卻對單元測試望而卻步。有些時候並不是不想寫,而是常常會碰到諸如不能模擬一次HTTP請求,不能讀取配置文件,測試類的構造參數太多等問題,讓開發者放下了碼字的腳步。這些問題確實存在,但它們阻止不了我們那顆要寫單元測試的心。單元測試的優點很... ...


在我們碼字過程中,單元測試是必不可少的。但在從業過程中,很多開發者卻對單元測試望而卻步。有些時候並不是不想寫,而是常常會碰到下麵這些問題,讓開發者放下了碼字的腳步:

  1. 這個類初始數據太麻煩,你看:new MyService(new User("test",1), new MyDAO(new Connection(......)),new ToManyPropsClass(......) .....) 。我:。。。
  2. 這個代碼內部邏輯都是和Cookie有關,我單元測試不好整啊,還是得啟動到瀏覽器里一個按鈕一個按鈕點。
  3. 這個代碼內部讀了配置文件,單元測試也不能給我整個配置文件啊?
  4. 這個代碼主要是驗證WebAPI入口得模型綁定,必須得調用一次啊?

這些問題確實存在,但它們阻止不了我們那顆要寫單元測試的心。單元測試的優點很多,你或許可以不管。但至少能讓你從那些需要在瀏覽器里點擊10多下的操作里解脫出來。本文從一個簡單的邏輯測試出發,慢慢拉開測試的大幕,讓你愛上測試。文章主要是傳播一些單元測試的理念,其次才是介紹asp.net core中的單元測試。

本文使用的環境為asp.net core 2.1 webapi,所有代碼都發佈到github:https://github.com/yubaolee/DotNetCoreUnitTestSamples 項目依賴的包為:

    <PackageReference Include="Microsoft.AspNetCore.TestHost" Version="2.1.0" />
    <PackageReference Include="Microsoft.NET.Test.Sdk" Version="15.5.0" />
    <PackageReference Include="Moq" Version="4.8.3" />
    <PackageReference Include="NUnit" Version="3.9.0" />
    <PackageReference Include="NUnit3TestAdapter" Version="3.9.0" />
    <PackageReference Include="System.Linq" Version="4.3.0" />

可以直接修改csproj文件,也可以nuget導入。

測試的業務邏輯為:

public class UserService{
        public bool CheckLogin(UserInfo user)
        {
            return user.Name == user.Password;  //登錄邏輯,為了看著舒服,少點
        }
    }
public class UserInfo{
        public string Name { get; set; }
        public string Password { get; set; }
    }

測試的WebAPI控制器如下:

 public class ValuesController : ControllerBase
    {
        private UserService _service;

        public ValuesController(UserService service)
        {
            _service = service;
        }

        [HttpGet]
        [Route("checklogin")]
        public bool CheckLogin([FromQuery]UserInfo user)
        {
            return _service.CheckLogin(user);
        }

    }

普通業務的單元測試

public class TestService
    {
        private UserService _service;

        [SetUp]
        public void Init()
        {
            var server = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
            _service = server.Host.Services.GetService<UserService>();
        }
        [Test]
        public void TestLogin()
        {
            bool result = _service.CheckLogin(new UserInfo { Name = "yubao", Password = "yubao" });
            Assert.IsTrue(result);
        }
    }

 在做業務測試過程中要善於使用註入功能,而不是使用new對象的方式,比如這裡的Host.Services.GetService,防止出現new MyService(new User("test",1), new MyDAO(new Connection(......)),new ToManyPropsClass(......) .....)這種尷尬。用的越多你就越能體會這種做法的好處。我在openauth.net中使用的是autofac的AutofacServiceProvider。

測試Controller

很多時候我們需要測試頂層的controller(八成是controller里混的有業務邏輯)。這時我們可以快速的寫出下麵的測試代碼:

 public class TestController
    {
        private ValuesController _controller;

        [SetUp]
        public void Init()
        {
            var server = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
            _controller = server.Host.Services.GetService<ValuesController>();
        }
        [Test]
        public void TestLogin()
        {
            bool result = _controller.CheckLogin(new UserInfo{Name = "yubao",Password = "yubao"});
            Assert.IsTrue(result);
        }
    }

這段代碼在JAVA spring mvc框架下是沒有問題的,但在asp.net core 中,你會發現:

獲取不到controller?spring mvc的理念就是萬物皆服務,哪怕是一個controller也是一個普通的服務。但微軟不喜歡這樣,預設時它要掌控controller的生死The Subtle Perils of Controller Dependency Injection in ASP.NET Core MVC 有人在聲討微軟了)。所以我們不能通過普通的ServicCollection來註入和獲取它,除非你指明Controller As Service,如下:

 public void ConfigureServices(IServiceCollection services)
        {
            services.AddMvc().AddControllersAsServices().SetCompatibilityVersion(CompatibilityVersion.Version_2_1);
        }

這時即可順利測試通過。

測試含有HTTP上下文的業務邏輯,比如Cookie、URL中的QueryString

在平時的代碼過程中,常常會和HTTP上下文HttpContext打交道,最常見的如request、response、cookie、querystring等,比如我們新的邏輯:

public class UserService
    {
        private IHttpContextAccessor _httpContextAccessor;

        public UserService(IHttpContextAccessor httpContextAccessor)
        {
            _httpContextAccessor = httpContextAccessor;
        }

        public bool IsLogin()
        {
            return _httpContextAccessor.HttpContext.Request.Cookies["username"] != null;
        }
    }

這時如何測試呢?馬丁福勒在他的大作《企業應用架構模式》中明確指出“測試樁”的概念,來應對這種情況。各種Mock框架應運而生。比如我最喜歡的Moq:

public class TestCookie
    {
        private UserService _service;

        [SetUp]
        public void Init()
        {
            var httpContextAccessorMock = new Mock<IHttpContextAccessor>();
            httpContextAccessorMock.Setup(x => x.HttpContext.Request.Cookies["username"]).Returns("yubaolee");

            var server = new TestServer(WebHost.CreateDefaultBuilder()
                .ConfigureServices(u =>u.AddScoped(x =>httpContextAccessorMock.Object))
                .UseStartup<Startup>());
            _service = server.Host.Services.GetService<UserService>();
        }
        [Test]
        public void TestLogin()
        {
            bool result = _service.IsLogin();
            Assert.IsTrue(result);
        }
    }

  測試一次HTTP請求

 有時我們需要測試Mvc框架的模型綁定,看看一次客戶端的請求是否能被正確解析,亦或者測試WebAPI入口的一些Filter AOP等是否被正確觸發,這時就需要測試一次HTTP請求。從嚴格意義上來講這種測試已經脫離的單元測試的範疇,屬於集成測試。但這種測試代碼可以節省我們大量的重覆勞動。asp.net core中可以通過TestServer快速實現這種模擬:

public class TestHttpRequest
    {
        private TestServer _testServer;

        [SetUp]
        public void Init()
        {
            _testServer = new TestServer(WebHost.CreateDefaultBuilder().UseStartup<Startup>());
        }
        [Test]
        public void TestLogin()
        {
            var client = _testServer.CreateClient();
            var result = client.GetStringAsync("/api/values/checklogin?name=yubao&password=yubao");
            Console.WriteLine(result.Result);
        }
    }

在進行單元測試的過程中,測試的理念(或者TDD的思維?)異常重要,它能幫助你構建和諧優美的代碼。

G


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 遇到這樣一個問題。我想要統計某個文件夾下有多少個py文件怎麼辦。 用python能解決嗎?答案,能。 解決辦法,使用glob 代碼如下: 很簡單的幾行代碼,。 第一步,導入glob庫 第二步,使用glob下麵的glob的方法,參數是路徑下判斷的文件 第三步:列印: 結果: 這個是和我當前目錄所有的p ...
  • 消息隊列已經逐漸成為企業IT系統內部通信的核心手段。它具有低耦合、可靠投遞、廣播、流量控制、最終一致性等一系列功能,成為非同步RPC的主要手段之一。 消息被處理的過程相當於流程A被處理。我們這裡以一個實際的模型來討論下,比如用戶下單成功時給用戶發簡訊,如果沒有這個消息隊列,我們會選擇同步調用發簡訊的接 ...
  • 作為一個比較喜歡出去見識世界、看看自然風光的人,這幾年有幸跑了一些地方,在祖國大地不同的地方見識了不同的人文風物,手機里也存了不少照片,想著如果以某種方式展現來回憶我的旅途的話,或許會給自己帶來不一樣的體驗。目標口號已經想好:去過的風景,直觀豐富地展示你的人生旅途,帶來新的回憶感受。在看到leafl ...
  • 考慮一下利用Python製作一個整蠱、木馬軟體,我提供思路。(清楚到沒學過編程的人也理解) 1、首先一個黑客做一個整蠱或者木馬軟體,一定不會讓你能夠關閉它。 2、裡面經常會附帶欺騙的方法。 3、最終實現某種目的。 前段時間在抖音上看到個有趣的程式,看樣子是VB寫的,首先就要用到欺騙的方法,利用軟體名 ...
  • 單例模式是java的一種設計模式,它是指在設計一個類時需要保證在整個程式運行期間針對該類值存在一個實例對象。例: class Single{ private static Single INSTANCE = new Single(); private Single() {} public stati ...
  • 原創 以下內容來自《Java 2實用教程》,主編:耿祥義、張躍平 鑒於面向抽象編程和麵向介面編程思維培養的重要性,寫此博客鞏固。 面向抽象編程: 在設計程式時,經常會使用到abstract類,其原因是,abstract類只關心操作,而不關心這些操作具體的實現細節, 可以使程式的設計者把主要精力放在程 ...
  • Android屏幕解鎖圖案利用Python破解 在 Android 手機上,我們可以通過設置鎖定圖案來當做密碼對手機鎖屏。 在 Android 存儲時使用的是明文轉換後採用散列方式存儲。 這種密碼有三個要求: 最少四個數 最多九個數 無重覆數 加密存儲過程如下: 第一步:隨便輸入一個圖形進行測試 可 ...
  • 第一步:頁面分析 我們要抓取的商品頁面: 這裡我們用正則表達式匹配出該段信息。進一步可以發現它是一個json數據,用json.loads()解析。解析後我們可以藉助線上工具可以查看數據的層級結構,找到需要數據的key。這裡我們把頁面分析與數據提取放到一塊來講了,代碼邏輯: 第三步:開始抓取 第四步: ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...