标签:
现在这个项目已经有阶段性的模块完成了,所以就想着对这些模块进行单元测试,以保证项目的代码的质量。首先虽然标题是mvc+webapi实质上我只是对mvc进行的测试。用的时候vs的unit test generator.至于它的安装和介绍在这不做详细介绍。好的现在开始总结我的单元测试总结。
此项目我是把它当作模块插件来处理。
[TestClass] public class mtnfim { //实例化测试mvc Controller mgtController mtn_fim = new mgtController(); [TestMethod] public void IndexTest() { //调用类中方法 var ret = mtn_fim.Index() as ViewResult; int errcode = testresult.action_result(ret).errcode; Assert.AreEqual(0, errcode); } }
Assert.AreEqual() 测试指定的值是否相等,如果相等,则测试通过;
AreSame() 用于验证指定的两个对象变量是指向相同的对象,否则认为是错误
AreNotSame() 用于验证指定的两个对象变量是指向不同的对象,否则认为是错误
Assert.IsTrue() 测试指定的条件是否为True,如果为True,则测试通过;
Assert.IsFalse() 测试指定的条件是否为False,如果为False,则测试通过;
Assert.IsNull() 测试指定的对象是否为空引用,如果为空,则测试通过;
Assert.IsNotNull() 测试指定的对象是否为非空,如果不为空,则测试通过;
我项目当中mvc Controller的返回类型有两个JsonResult。ActionrResult.两种,所以针对于assert类的使用显然我要对mvc的返回的对象进行处理。
jsonresult的处理方式:
通过nugt下载ExposedObject把 对象转为dynami然后对其对象进行操作。代码如下。
var jsonret = mtn_fim.add_feeitem("测试", "cs"); var ret = Exposed.From(jsonret.Data); int errcode = ret.errcode; Assert.AreEqual(6002, errcode);
ActionrResult的处理:
相对与jsonresult直接操作返回对象即可,对此我写了一个方法。
public static class testresult { public static result<object> action_result(ViewResult ret) { return ((wx.web.result<object>)(((System.Web.Mvc.ViewResultBase)((ret))).Model)); } public static result apicitroller_result(System.Web.Http.IHttpActionResult action) { var rs = ((System.Web.Http.Results.OkNegotiatedContentResult<wx.web.result>) (action)).Content; return rs; } }
使用如下:
var ret = mtn_fim.Index() as ViewResult; int errcode = testresult.action_result(ret).errcode; Assert.AreEqual(0, errcode);
3.实现之外的事:
3.1.数据库初始化
以上已经初步实现了单元测试的基本实现,但是考虑到单元测试重复测试,我用ef code first动态的数据库创建,和需要测试的数据。
public void Initialize(FeeDbContext context) { List<organization> orgs = new List<organization>() { new organization(){oid=1, name="新中新集团1", state=true,reg_time=DateTime.Now}, new organization(){oid=2, name="山东大学", state=true,reg_time=DateTime.Now}, new organization(){oid=3, name="浙江大学", state=true,reg_time=DateTime.Now}, new organization(){oid=4, name="同济大学", state=true,reg_time=DateTime.Now} }; List<fee_item> fim = new List<fee_item>() { new fee_item(){name="电费",code="pow",type= feeItemType.Normal,State=true,Icon="img/jf_dianf.png",sort=0,appid=0,reg_time=DateTime.Now}, new fee_item(){name="网费",code="net",type= feeItemType.Normal,State=true,Icon="img/jf_wangf.png",sort=1,appid=1,reg_time=DateTime.Now}, new fee_item(){name="四六级",code="cet",type= feeItemType.Normal,State=true,Icon="img/jf_siliuj.png",sort=2,appid=2,reg_time=DateTime.Now}, new fee_item(){name="报名费",code="pow",type= feeItemType.Normal,State=true,Icon="img/jf_baomingf.png",sort=3,appid=3,reg_time=DateTime.Now}, new fee_item(){name="保险费",code="pow",type= feeItemType.Normal,State=false,Icon="img/jf_baox.png",sort=4,appid=4,reg_time=DateTime.Now}, new fee_item(){name="住宿费",code="pow",type= feeItemType.Normal,State=true,Icon="img/jf_dianf.png",sort=5,appid=5,reg_time=DateTime.Now} }; List<school_power> con_power = new List<school_power>() { new school_power(){oid=1,power_code="ykt",power_id="1",reg_time=DateTime.Now}, new school_power(){oid=1,power_code="sims",power_id="2",reg_time=DateTime.Now} }; //depts.ForEach(o => context.dept.Add(o)); orgs.ForEach(o => context.organization.Add(o)); fim.ForEach(o => context.fee_item.Add(o)); con_power.ForEach(o => context.control_power.Add(o)); context.SaveChanges(); }
public void initial() { try { var context = new FeeDbContext(); if (context.Database.Exists()) { context.Database.Delete(); } new List<IDataInitializer<FeeDbContext>>() { new DataInit4dept() }.Setup<FeeDbContext>(context); } catch (DbEntityValidationException ex) { StringBuilder error = new StringBuilder(); foreach (var item in ex.EntityValidationErrors) { foreach (var item2 in item.ValidationErrors) { error.Append(string.Format("{0}:{1}\r\n", item2.PropertyName, item2.ErrorMessage)); } } Console.WriteLine("数据库初始化报错:" + error); throw ex; } catch (Exception e) { Console.WriteLine("数据库初始化报错:" + e.Message); throw e; } }
ef code first写好之后就又到单元测试的部分了,所以要想每次执行单元测试内容的时候都执行一下以上内用就要用到单元测试的附属属性:
[ClassInitialize()] 在运行类的第一个测试前先运行代码
[ClassCleanup()] 在运行完类中的所有测试后再运行代码
[TestInitialize()] 在运行每个测试前先运行代码
[TestCleanup()] 在运行完每个测试后运行代码
[ClassInitialize] public static void initialdb(TestContext context) { //初始化数据库 new DataBuilder().initial(); //加载webapi bin文件 apiconfig_factory.Instance.Load_Apis(); }
直到这单元测试的实现也就算结束了,但是对于单元测试还有很多东西没有往上写,比如测试模型,覆盖率等等,对于这个项目的测试也不仅限于单元测试,后边的性能测试和ui测试都会抽时间进行总结。
标签:
原文地址:http://www.cnblogs.com/kmonkeywyl/p/5369189.html