在线时间:8:00-16:00
迪恩网络APP
随时随地掌握行业动态
扫描二维码
关注迪恩网络微信公众号
首先要啰嗦几句。 单元测试是TDD的重要实践方法,也是代码质量的一种保证手段。在项目的工程化开发中,研发人员应该尽量保证书写Unit Test,即使不使用TDD。 (VS中,我们可以直接使用微软提供的一套单元测试框架,一般使用足够了,特别需求的话,可以使用其他更好的框架。) 书写单元测试时,我们并不一定真的要去连接数据库,毕竟就算只使用自己计算机上的研发数据库,也不能保证数据正确性和完备性,毕竟自己经常会操作些垃圾数据。 这个时候就需要模拟一个“数据库”来构造我们想要的一些数据。这个就是Mock最直接的需求。然后当我们进一步实践,会发现我们的Service层等,也可以用Mock模拟对象,而不是非要去new一个真实对象。真实场景是当我们研发小组合作时,你作为更高层的研发人员,可能只拿到了服务层的接口,具体的实现类你的同事还在研发中,这个时候你要做Unit Test,就只有模拟一个假的Service实现了。 C#的单元测试框架中,有一套Mock框架就叫Moq。 Moq可以直接在VS 2013及以后的版本中通过Nuget获取。以前的版本的VS可以到github上下载Moq的dll。 Moq的github地址为:Moq Moq的QuickStart页面为:QuickStart 深入学习,可以直接看此文档。 MVC中,最直接需要模拟的应该就是HttpContext相关对象,如HttpRequest、Server、Session等对象。以HttpRequest为例。 首先,我们要知道, controller中相关HttpContext的对象是ControllerContext,它就是HttpContextBase。模拟的HttpContext通过它绑定给Controller。 controller.ControllerContext = new ControllerContext(mockContext.Object, new RouteData(), controller); mockContext.Object就是我们用Moq模拟的HttpContextBase。 这里是绑定的代码,倒推回去,我们应该先生成mockContext,如下: private Mock<HttpContextBase> mockContext = new Mock<HttpContextBase>(); 实际我们代码可能使用的是Request["xxx"]、Session["yyy"],这些对象又依赖于HttpContextBase,所以我们需要模拟它们,然后绑定到 mockContext。如下 var mockRequest = new Mock<HttpRequestBase>(); //模拟Request["xxx"] if (dataIndexed != null) { foreach (var pair in dataIndexed) { mockRequest.Setup(x => x[pair.Key]).Returns(pair.Value); } } mockContext.SetupGet(x => x.Request).Returns(mockRequest.Object); 注意最后一句就是将模拟的Request绑定到模拟的HttpContextBase上。代码含义是:当通过mockContext.Request(即它的get方法)得到Request对象时,把mockRequest模拟的HttpRequest对象返回。 Moq的方法都是比较直白的含义,如上就是:SetupGet(x => x.Request).Returns($$$),针对对象Request使用Get方法时,Returns相应的对象$$$。其他的对象,不管是HttpContextBase的还是Request的再子层对象,都通过这样的方法设置,前提是相应的类中有此属性(get,set)。 回过头继续看上面代码这段mockRequest.Setup(x => x[pair.Key]).Returns(pair.Value); 这个就是直接针对值进行设置,这里是针对类的Indexedr进行设置。 还有一些其他方法,需要时就看QuickStart了。 另外,针对全局的HttpContext对象,在单元测试中,它是null的,所以为了保证单元测试可进行,需要对其进行包装,在项目中使用包装的类进行访问。这样,单元测试时,就注入自己模拟的HttpContext对象。然而HttpContext是sealed类,是不能被Mock的。所以我们可以在包装类中,使用两个对象,分别指向Mock的对象和真实的HttpContext,依据是否模拟的判断在代码中选择调用。也可以使用HttpContextWrapper来包装HttpContext,因为HttpContextWrapper是HttpContextBase的实现。如: /// <summary> /// 全局HttpContext的包装类,以便单元测试 /// </summary> public class CmsHttpContext { /// <summary> /// 当前单例对象 /// </summary> private static CmsHttpContext _instance = new CmsHttpContext(); /// <summary> /// 包装的HttpContext /// </summary> private static HttpContextBase wrapper = null; /// <summary> /// 是否被包装 /// </summary> private static bool IsWrap = false; private CmsHttpContext() { } /// <summary> /// 当前HttpContext对象 /// </summary> public static HttpContextBase Current { get { if (!IsWrap) { wrapper = new HttpContextWrapper(HttpContext.Current); } return wrapper; } } /// <summary> /// 包装外部 HttpContext,仅用于单元测试中 /// </summary> /// <param name="context"></param> public static void Wrap(HttpContextBase context) { IsWrap = true; wrapper = context; } } 其他要注意的点: 1.Action的方法直接调用即可以执行。针对ViewBag.XXX,使用Controller对象调用,如mController.ViewBag.XXX 2.JsonResult中Json(XXX),如果XXX是动态类型的话,它在传输后会变成object,单元测试中无法识别它相应的属性,可以使用框架ExposedObject(Nuget中可以直接下载)进行包装,将其包装回dynamic进行测试,如下: var jsonData = Exposed.From(result.Data); Assert.IsTrue(jsonData.total > 0); Assert.IsTrue(jsonData.list.Count > 0);
|
请发表评论