[转] 如何让代码可测试化(C#)
让代码可测试化
本篇介绍如何把我们目前最常见的代码转换为可以单元测试的代码,针对业务逻辑层来实现可测试性,我们以银行转账为例,通常代码如下:
|
public class TransferController { private TransferDAL dal = new TransferDAL(); public bool TransferMoney(string fromAccount, string toAccount, decimal money) { //验证:比如账号是否存在、账号中是否有足够的钱用来转账 if (fromAccount == null || fromAccount.Trim().Length == 0) return false; if (toAccount == null || toAccount.Trim().Length == 0) return false; if (IsExistAccount(fromAccount))//检查from账号是否存在 return false; if (IsAccountHasEnoughMoney(fromAccount))//检查from账号中是否有足够的钱用来转账 return false; //更新数据库 dal.TransferMoney(fromAccount, toAccount, money); //发送邮件 EmailSender.SendEmail("aaa@aa.com", "xxxxxxxx", "yyyyyyyyyy"); return true; } private bool IsAccountHasEnoughMoney(string fromAccount) { throw new System.NotImplementedException(); } private bool IsExistAccount(string fromAccount) { throw new System.NotImplementedException(); } } |
|
相应sql语句如下: public void TransferMoney(string fromAccount, string toAccount, decimal money) { string sql = @" UPDATE Accounts SET Money=Money-@Money WHERE Account=@FromAccount UPDATE Accounts SET Money=Money+@Money WHERE Account=@FromAccount "; } |
扎眼一看,这转账操作的逻辑写在了sql语句中(没有弱化外部操作),这样就会导致对业务逻辑代码的不可测试性,因此需要重构转账的计算部分,改成如下:
|
public class TransferController { private TransferDAL dal = new TransferDAL(); public bool TransferMoney(string fromAccount, string toAccount, decimal money) { //验证:比如账号是否存在、账号中是否有足够的钱用来转账 if (fromAccount == null || fromAccount.Trim().Length == 0) return false; if (toAccount == null || toAccount.Trim().Length == 0) return false; if (IsExistAccount(fromAccount))//检查from账号是否存在 return false; if (IsAccountHasEnoughMoney(fromAccount))//检查from账号中是否有足够的钱用来转账 return false; //更新数据库 using(TransactionScope ts=new TransactionScope()) { dal.MinuseMoney(fromAccount, money); dal.PlusMoney(toAccount, money); ts.Complete(); } //发送邮件 EmailSender.SendEmail("aaa@aa.com", "xxxxxxxx", "yyyyyyyyyy"); return true; } private bool IsAccountHasEnoughMoney(string fromAccount) { throw new System.NotImplementedException(); } private bool IsExistAccount(string fromAccount) { throw new System.NotImplementedException(); } } |
相对于业务逻辑层来说,分析出外部接口有:邮件发送、数据访问对象,因此增加这2个接口到代码中,变成如下:
|
public class TransferController { private ITransferDAO dao = new TransferDAL(); private IEmailSender emailSender=new XXXXXXXXXXXXXXX();//由于一般的email发送类都是static的,不能new,这部分先留着,等下一步解决 public bool TransferMoney(string fromAccount, string toAccount, decimal money) { //验证:比如账号是否存在、账号中是否有足够的钱用来转账 if (fromAccount == null || fromAccount.Trim().Length == 0) return false; if (toAccount == null || toAccount.Trim().Length == 0) return false; if (IsExistAccount(fromAccount))//检查from账号是否存在 return false; if (IsAccountHasEnoughMoney(fromAccount))//检查from账号中是否有足够的钱用来转账 return false; //更新数据库 using(TransactionScope ts=new TransactionScope()) { this.dao.MinuseMoney(fromAccount, money); this.dao.PlusMoney(toAccount, money); ts.Complete(); } //发送邮件 this.emailSender.SendEmail("aaa@aa.com", "xxxxxxxx", "yyyyyyyyyy"); return true; } private bool IsAccountHasEnoughMoney(string fromAccount) { throw new System.NotImplementedException(); } private bool IsExistAccount(string fromAccount) { throw new System.NotImplementedException(); } } |
但是此时的2个接口,实际系统运行过程中还是会强耦合2个具体类,还是不可测试,怎么办呢?利用构造函数注入:
|
public class TransferController { private ITransferDAO dao; private IEmailSender emailSender; public TransferController()//实际运行时可以用这个构造 { dao = new TransferDAL(); emailSender = new EmailSenderAgent(); } public TransferController(ITransferDAO dao, IEmailSender emailSender)//测试时用这个构造注入Fake对象 { this.dao = dao; this.emailSender = emailSender; } public bool TransferMoney(string fromAccount, string toAccount, decimal money) { //验证:比如账号是否存在、账号中是否有足够的钱用来转账 if (fromAccount == null || fromAccount.Trim().Length == 0) return false; if (toAccount == null || toAccount.Trim().Length == 0) return false; if (IsExistAccount(fromAccount))//检查from账号是否存在 return false; if (IsAccountHasEnoughMoney(fromAccount))//检查from账号中是否有足够的钱用来转账 return false; //更新数据库 using(TransactionScope ts=new TransactionScope()) { this.dao.MinuseMoney(fromAccount, money); this.dao.PlusMoney(toAccount, money); ts.Complete(); } //发送邮件 this.emailSender.SendEmail("aaa@aa.com", "xxxxxxxx", "yyyyyyyyyy"); return true; } private bool IsAccountHasEnoughMoney(string fromAccount) { throw new System.NotImplementedException(); } private bool IsExistAccount(string fromAccount) { throw new System.NotImplementedException(); } } |
终于,可以编写单元测试了,看下面:
|
class TransferMoneyTest { public void TransferMoney_Validate_FromAccount_Null_Test() { string fromAccount=null; string toAccount="bbbbbbbbbbbb"; decimal money=100; TransferController ctl = new TransferController(null, null);//因为这个测试用不到这2个接口,所以用了null bool real= ctl.TransferMoney(fromAccount, toAccount, money); Assert.IsFalse(real); } public void TransferMoney_Validate_FromAccount_Empty_Test() { string fromAccount = ""; string toAccount = "bbbbbbbbbbbb"; decimal money = 100; TransferController ctl = new TransferController(null, null);//因为这个测试用不到这2个接口,所以用了null bool real = ctl.TransferMoney(fromAccount, toAccount, money); Assert.IsFalse(real); } public void TransferMoney_Validate_FromAccount_AllSpace_Test() { string fromAccount = " "; string toAccount = "bbbbbbbbbbbb"; decimal money = 100; TransferController ctl = new TransferController(null, null);//因为这个测试用不到这2个接口,所以用了null bool real = ctl.TransferMoney(fromAccount, toAccount, money); Assert.IsFalse(real); } public void TransferMoney_Validate_FromAccount_NotExist_Test() { string fromAccount = "11111111111111"; string toAccount = "bbbbbbbbbbbb"; decimal money = 100; ITransferDAO dao = new FakeTransferDAO_NullAccount(); TransferController ctl = new TransferController(dao, null);//因为这个测试用不到IEmailSender接口,所以用了null bool real = ctl.TransferMoney(fromAccount, toAccount, money); Assert.IsFalse(real); } public void TransferMoney_Validate_FromAccount_NotEnoughMoney_Test() { string fromAccount = "11111111111111"; string toAccount = "bbbbbbbbbbbb"; decimal money = 100; ITransferDAO dao = new FakeTransferDAO_NotEnoughMoney(); TransferController ctl = new TransferController(dao, null);//因为这个测试用不到IEmailSender接口,所以用了null bool real = ctl.TransferMoney(fromAccount, toAccount, money); Assert.IsFalse(real); } } |
|
用到了如下2个Fake类 class FakeTransferDAO_NullAccount : ITransferDAO { public void MinuseMoney(string fromAccount, decimal money) { throw new NotImplementedException(); } public void PlusMoney(string toAccount, decimal money) { throw new NotImplementedException(); } public Account GetAccount(string accountId) { return null; } } class FakeTransferDAO_NotEnoughMoney: ITransferDAO { public void MinuseMoney(string fromAccount, decimal money) { throw new NotImplementedException(); } public void PlusMoney(string toAccount, decimal money) { throw new NotImplementedException(); } public Account GetAccount(string accountId) { Account account = new Account(); account.Money = 20; return account; } } |
暂时先写到这里,呵呵...
[转] 如何让代码可测试化(C#)的更多相关文章
- 测试化工具XCTestCase
layout: post title: "Xcode 7智能测试化工具XCTest学习" subtitle: "Xcode 7智能测试化工具XCTest学习" ...
- 来自ebay内部的「软件测试」学习资料,覆盖GUI、API自动化、代码级测试及性能测试等,Python等,拿走不谢!...
在软件测试领域从业蛮久了,常有人会问我: 刚入测试一年,很迷茫,觉得没啥好做的-- 测试在公司真的不受重视,我是不是去转型做开发会更好? 资深的测试架构师的发展路径是怎么样的?我平时该怎么学习? 我 ...
- mvn编写主代码与测试代码
maven编写主代码与测试代码 3.2 编写主代码 项目主代码和测试代码不同,项目的主代码会被打包到最终的构件中(比如jar),而测试代码只在运行测试时用到,不会被打包.默认情况下,Maven假设项目 ...
- js代码如何测试代码运行时间
function add(){ //这里放要执行的代码 } //开始测试并输出 function test() { var start=new Date().getTime(); add(); var ...
- Java代码安全测试解决方案
Java代码安全测试解决方案: http://gdtesting.com/product.php?id=106
- maven编写主代码与测试代码
3.2 编写主代码 项目主代码和测试代码不同,项目的主代码会被打包到最终的构件中(比如jar),而测试代码只在运行测试时用到,不会被打包.默认情况下,Maven假设项目主代码位于src/main/ja ...
- R︱Rstudio 1.0版本尝鲜(R notebook、下载链接、sparkR、代码时间测试profile)
每每以为攀得众山小,可.每每又切实来到起点,大牛们,缓缓脚步来俺笔记葩分享一下吧,please~ --------------------------- 2016年11月1日,RStudio 1.0版 ...
- 使用Jenkins结合Gogs和SonarQube对项目代码进行测试、部署、回滚,以及使用keepalived+haproxy调度至后端tomcat
0 环境说明 主tomcat:192.168.0.112 备tomcat:192.168.0.183 haproxy+keepalived-1:192.168.0.156 haproxy+keepal ...
- Tars | 第7篇 TarsJava Subset最终代码的测试方案设计
目录 前言 1. SubsetConf配置项的结构 1.1 SubsetConf 1.2 RatioConfig 1.3 KeyConfig 1.4 KeyRoute 1.5 SubsetConf的结 ...
随机推荐
- “全栈2019”Java第一章:安装JDK11(Mac)
难度 初级 学习时间 10分钟 适合人群 零基础 开发语言 Java 开发环境 JDK v11 文章原文链接 “全栈2019”Java第一章:安装JDK11(Mac) 下一章 “全栈2019”Java ...
- BZOJ 4719--天天爱跑步(LCA&差分)
4719: [Noip2016]天天爱跑步 Time Limit: 40 Sec Memory Limit: 512 MBSubmit: 1464 Solved: 490[Submit][Stat ...
- python爬虫的页面数据解析和提取/xpath/bs4/jsonpath/正则(1)
一.数据类型及解析方式 一般来讲对我们而言,需要抓取的是某个网站或者某个应用的内容,提取有用的价值.内容一般分为两部分,非结构化的数据 和 结构化的数据. 非结构化数据:先有数据,再有结构, 结构化数 ...
- Wireshark系列(从入门到精通的10个干货)
Wireshark(前称Ethereal)是一个网络封包分析软件.网络封包分析软件的功能是撷取网络封包,并尽可能显示出最为详细的网络封包资料.Wireshark使用WinPCAP作为接口,直接与网卡进 ...
- bash脚本编程学习笔记(二)
1.脚本编程之函数 函数是实现结构化编程重要的思想,主要目的是实现代码重用 定义一个函数: function FUNCNAME { command //函数体 } FUNCNAME(){ //函数 ...
- JSP常用Form表单控件
[easyui]--combobox--赋值和获取选中的值 /初始化下拉选框 $('#communityIdDiv').combobox({ url:basepath+"pushContro ...
- leetcode-39-组合总和(有趣的递归)
题目描述: 给定一个无重复元素的数组 candidates 和一个目标数 target ,找出 candidates 中所有可以使数字和为 target 的组合. candidates 中的数字可以无 ...
- springboot自定义jmx对象
在使用springboot-admin对springboot项目进行监控的时候我们发现其是具有web访问jmx对象的功能的,那它内部是怎么实现的呢. Jolokia是一个JMX-http桥梁,它提供了 ...
- scrapyd的安装
.安装 pip3 install scrapyd 二.配置 安装完毕之后,需要新建一个配置文件/etc/scrapyd/scrapyd.conf,Scrapyd在运行的时候会读取此配置文件. 在Scr ...
- MySQL 连接注意事项
外连接 A LEFT JOIN B join_condition 数据表B的结果集依赖数据表A 数据表A的结果集根据左连接条件依赖所有数据表 B表除外). 左外连接条件决定如何检索数据表B(在没有指定 ...