.NET单元测试(二):入门

基于状态测试

  在上一篇文章中,我们举了一个带返回值的例子,那么无返回值的情况下又该怎样写单元测试呢?

有如下代码:

public IList<string> Names = new List<string>();
  
public void Reset()
{
    Names.Clear();
}

我们发现,Reset方法内部执行的是Names列表的清空操作,这个操作可以抽象成对被测试类状态的更改,要验证状态更改是否符合预期,我们只需要验证更改前后是否符合预期即可。在这里,只需要测试Reset方法是否按照我们预期的把Names清空即可。如下:

/// <summary>
/// 条件:Names不为空
/// 预期:清空Names
/// </summary>
[TestMethod()]
public void ResetTest_NamesNotEmpty_NamesEmpty()
{
   //Arrange
   var document = new Document();
   document.Names.Add("name0");
   document.Names.Add("name1");

   //Action
   document.Reset();

   //Assert
   Assert.AreEqual(document.Names.Count, 0);
}

依赖外部对象的测试

  单元测试需要能够快速独立运行,隔离掉对外部的依赖是非常必要的,比如文件系统、硬件数据、web服务等。

如下代码:

///<summary>
/// 判断当前字符串是否是合法的html字符串
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public bool IsValidHtml(string input)
{
    var textService = new TextService();
 
    return textService.IsValidHtml(input);
}

可以看到,当前方法依赖TextService来验证html,但是在运行单元测试时,TextService的状态是未知的,它甚至可能还未开发完成。因此,需要隔离掉对TextService的依赖。

而TextService是在IsValidHtml方法内部创建的,我们无法隔离,这个时候就需要对方法进行一系列的修改,以使得它达到可测试的要求(这就是所谓的单元测试约束设计)。

再进一步的分析,可以发现依赖的是TextService提供的IsValidHtml()方法,而并非TextService这个对象,这就好说了,让IsValidHtml()依赖可以提供html验证的接口,我们就可以不用依赖TextService这个对象了,我们抽取接口:

public interface ITextService
{
    bool IsValidHtml(string input);
}

这样我们就可以从对具体实现的依赖解耦为对接口的依赖,因此,在测试方法中我们可以很方便的用一个假的ITextService的实现来替代真实的TextService,由此隔离对真实外部服务的依赖。

这个假的ITextService的实现我们称为 伪对象

如下SubTextService就是我们的伪对象:

public class SubTextService : ITextService
{
    private bool _isValidHtml;
 
    public void SetIsValidHtml(bool value)
    {
        _isValidHtml = value;
    }
 
    public bool IsValidHtml(string input)
    {
        return _isValidHtml;
    }
}

有了伪对象,怎么使用起来呢?

接下来介绍几种伪对象注入的方式

  • 构造函数注入

    这种方式需要被测试类提供一个带有ITextService参数的构造函数,我们修改被测试类:
    public Document(ITextService textService)
    {
        _textService = textService;
    }
    
    /// <summary>
    /// 判断当前字符串是否是合法的html字符串
    /// </summary>
    /// <param name="input"></param>
    /// <returns></returns>
    public bool IsValidHtml(string input)
    {
        return _textService.IsValidHtml(input);
    }
    
    接下来,在测试方法中就可以将伪对象注入进去了:
    /// <summary>
    /// 条件:传入Empty的字符串
    /// 预期:返回False
    /// </summary>
    [TestMethod()]
    public void IsValidHtml_EmptyInput_ReturnFalse()
    {
        //Arrange
        var subTextService = new SubTextService();
        subTextService.SetIsValidHtml(false);
        var document = new Document(subTextService);
    
        //Action
        var result = document.IsValidHtml(string.Empty);
    
        //Assert
        Assert.IsFalse(result);
    }
    
    这种方法比较简单,被测试类的代码改动也不大。
    但是,如果方法中依赖多个外部接口,需要构造函数的参数列表可能很长;或者被测试类中不同方法依赖了不同的外部接口,那么需要增加多个构造函数。
    因此,此方法需要根据情况谨慎使用。
  • 属性注入

    这种方式指的是被测试类将外部接口的依赖设计成可以公开属性:
    public ITextService TextService { get; set; }
    
    这样在单元测试中就可以方便的将伪对象注入进去。
    这种方法简单,对被测试类改动小。
    但是,将TextService设计成属性,会给外部一种TextService的赋值非必需的误解,然而在我们的设计中TextService是必须的。
    因此,不推荐使用。
  • 工厂注入

    工厂注入指的是当我们依赖的第三方接口是用工厂新建时,通过给工厂中注入伪对象来隔离对真实对象的依赖。
    public static class TextServiceFactory
    {
        private static ITextService _textService = new TextService();
    
        public static ITextService Create()
        {
            return _textService;
        }
    
        public static void SetTextService(ITextService textService)
        {
            _textService = textService;
        }
    }
    
    这种方法也比较简单,需要对工厂方法进行修改,改动量也不大。
    可根据情况使用。
  • 派生类注入

    派生类注入指的是在设计的时候,把对外部的依赖对象的获取设计成可以被继承,这样伪对象就可以在不修改原来代码的情况下完成注入:
    protected virtual ITextService GetTextService()
    {
        return new TextService();
    }
    
    写单元测试的时候,只需要用伪对象继承被测试类,就可以在重写GetTextService时,注入伪对象。
    //Document为被测试类
    public class SubDocument : Document
    {
        protected override ITextService GetTextService()
        {
            return new SubTextService();
        }
    }
    
    在单元测试时,就直接使用SubDocument即可.
    这种方法比较简单,而且不需要修改被测试类代码。
    推荐此方法。

  写单元测试可以为我们的代码增加一层保护,在设计程序时考虑单元测试也可以优化我们的设计,好处多多,何乐而不为呢(●'◡'●)



2017-3-17 23:29:07

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,132评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,802评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,566评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,858评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,867评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,695评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,064评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,705评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,915评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,677评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,796评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,432评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,041评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,992评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,223评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,185评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,535评论 2 343

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,598评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,577评论 18 399
  • 我们以神的名义 进行一场无关紧要的爱情 只是快乐常常被忘记 悲伤也难以想起 忍住不说的言语 随着时光逝去 在顶峰的...
    洛家仁人阅读 137评论 0 0
  • 到了该交文章的最后期限了,可是我还在试图找灵感,希望能做到字如泉涌般的自然流露出来。越是刻意,越是没有感觉。...
    布二阅读 146评论 0 0
  • 目的:实现自定义的上下拉刷新动画,尽可能少的代码侵入性。 轮子:MJRefresh 大多数方案:继承MJRefre...
    ccSundayChina阅读 1,490评论 3 14