当创建一个使用的类的单元测试HttpContext.Current.Cache
类,我使用NUnit时得到一个错误.功能是基本的 - 检查项目是否在缓存中,如果没有,则创建它并将其放入:
if (HttpContext.Current.Cache["Some_Key"] == null) { myObject = new Object(); HttpContext.Current.Cache.Insert("Some_Key", myObject); } else { myObject = HttpContext.Current.Cache.Get("Some_Key"); }
当从单元测试中调用它NullReferenceException
时,它在遇到第一Cache
行时失败.在Java中,我会使用Cactus来测试服务器端代码.我可以使用类似的工具用于C#代码吗? 这个问题提到了模拟框架 - 这是我测试这些方法的唯一方法吗?是否有类似的工具来运行C#测试?
此外,我不检查是否Cache
为null,因为我不想专门为单元测试编写代码,并假设它在服务器上运行时始终有效.这是有效的,还是应该在缓存周围添加空检查?
这样做的方法是避免直接使用HttpContext或其他类似的类,并用模拟替换它们.毕竟,你不是试图测试HttpContext是否正常运行(这是微软的工作),你只是试图测试方法是否应该被调用.
步骤(如果您只想了解技术而不需要挖掘大量博客):
创建一个界面,描述您希望在缓存中使用的方法(可能是GetItem,SetItem,ExpireItem等).称之为ICache或任何你喜欢的
创建一个实现该接口的类,并将方法传递给真正的HttpContext
创建一个实现相同接口的类,就像模拟缓存一样.如果您关心保存对象,它可以使用Dictionary或其他东西
更改原始代码,使其根本不使用HttpContext,而只使用ICache.然后代码将需要获取ICache的实例 - 您可以在类构造函数中传递一个实例(这就是依赖注入的真正原因),或者将其粘贴到某个全局变量中.
在您的生产应用程序中,将ICache设置为您真正的HttpContext-Backed-Cache,并在单元测试中将ICache设置为模拟缓存.
利润!
我同意其他人使用界面是最好的选择,但有时改变现有系统是不可行的.这里有一些代码,我只是从我的一个项目中混合在一起,它可以为您提供所需的结果.这是漂亮的或最好的解决方案中最远的东西,但如果你真的无法改变你的代码那么它应该完成工作.
using System; using System.IO; using System.Reflection; using System.Text; using System.Threading; using System.Web; using NUnit.Framework; using NUnit.Framework.SyntaxHelpers; [TestFixture] public class HttpContextCreation { [Test] public void TestCache() { var context = CreateHttpContext("index.aspx", "http://tempuri.org/index.aspx", null); var result = RunInstanceMethod(Thread.CurrentThread, "GetIllogicalCallContext", new object[] { }); SetPrivateInstanceFieldValue(result, "m_HostContext", context); Assert.That(HttpContext.Current.Cache["val"], Is.Null); HttpContext.Current.Cache["val"] = "testValue"; Assert.That(HttpContext.Current.Cache["val"], Is.EqualTo("testValue")); } private static HttpContext CreateHttpContext(string fileName, string url, string queryString) { var sb = new StringBuilder(); var sw = new StringWriter(sb); var hres = new HttpResponse(sw); var hreq = new HttpRequest(fileName, url, queryString); var httpc = new HttpContext(hreq, hres); return httpc; } private static object RunInstanceMethod(object source, string method, object[] objParams) { var flags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic; var type = source.GetType(); var m = type.GetMethod(method, flags); if (m == null) { throw new ArgumentException(string.Format("There is no method '{0}' for type '{1}'.", method, type)); } var objRet = m.Invoke(source, objParams); return objRet; } public static void SetPrivateInstanceFieldValue(object source, string memberName, object value) { var field = source.GetType().GetField(memberName, BindingFlags.GetField | BindingFlags.NonPublic | BindingFlags.Instance); if (field == null) { throw new ArgumentException(string.Format("Could not find the private instance field '{0}'", memberName)); } field.SetValue(source, value); } }
HttpContext.Current = new HttpContext(new HttpRequest(null, "http://tempuri.org", null), new HttpResponse(null));
如果您使用的是.NET 3.5,则可以在应用程序中使用System.Web.Abstractions.
Justin Etheredge 在如何模拟HttpContext(包含缓存类)方面有一篇很棒的文章.
从Justin的例子中,我使用HttpContextFactory.GetHttpContext将HttpContextBase传递给我的控制器.在模拟它们时,我只是构建一个模拟来调用缓存对象.
有一种更新的方法可以帮助在单元测试中专门处理Cache.
我建议使用微软新的MemoryCache.Default方法.您将需要使用.NET Framework 4.0或更高版本,并包含对System.Runtime.Caching的引用.
请参阅此处的文章 - > http://msdn.microsoft.com/en-us/library/dd997357(v=vs.100).aspx
MemoryCache.Default适用于Web和非Web应用程序.因此,您的想法是更新您的webapp以删除对HttpContext.Current.Cache的引用,并将其替换为对MemoryCache.Default的引用.稍后,当您运行决定单元测试这些相同的方法时,缓存对象仍然可用,并且不会为空.(因为它不依赖于HttpContext.)
这样您甚至不一定需要模拟缓存组件.