Microsoft.NET 解决方案,项目开发必知必会。

从这篇文章开始我将分享一系列我认为在实际工作中很有必要的一些.NET项目开发的核心技术点,所以我称为必知必会。尽管这一系列是使用.NET/C#来展现,但是同样适用于其他类似的OO技术平台,这些技术点可能称不上完整的技术,但是它是经验的总结,是掉过多少坑之后的觉醒,所以有必要花几分钟时间记住它,在真实的项目开发中你就知道是多么的有帮助。好了,废话不说了,进入主题。

我们在开发服务时为了调试方便会在本地进行一个基本的模块测试,你也可以认为是集成测试,只不过你的测试用例不会覆盖到80%以上,而是一些我们认为在开发时不是很放心的点才会编写适当的用例来测试它。

集成测试用例通常有多个执行上下文,对于我们开发人员来说我们的执行上下文通常都在本地,测试人员的上下文在测试环境中。开发人员的测试用来是不能够连接到其他环境中去的(当然视具体情况而定,有些用例很危险是不能够乱连接的,本文会讲如何解决),开发人员运行的集成测试用例所要访问的所有资源、服务都是在开发环境中的。这里依然存在但是,但是为了调试方便,我们还是需要能够在必要的时候连接到其他环境中去调试问题,为了能够真实的模拟出问题的环境、可真实的数据,我们需要能有一个这样的机制,在需要的时候我能够打开某个设置让其能够切换集成测试运行的环境上下文,其实说白了就是你所要连接的环境、数据源的连接地址。

本篇文章我们将通过一个简单的实例来了解如何简单的处理这中情况,这其实基于对测试用来不断重构后的效果。

 using System;
using Microsoft.VisualStudio.TestTools.UnitTesting; namespace OrderManager.Test
{
using ProductService.Contract; /// <summary>
/// Product service integration tests.
/// </summary>
[TestClass]
public class ProductServiceIntegrationTest
{
/// <summary>
/// service address.
/// </summary>
public const string ServiceAddress = "http://dev.service.ProductService/"; /// <summary>
/// Product service get product by pid test.
/// </summary>
[TestMethod]
public void ProductService_GetProductByPid_Test()
{
var serviceInstance = ProductServiceClient.CreateClient(ServiceAddress);
var testResult = serviceInstance.GetProductByPid(); Assert.AreNotEqual(testResult, null);
Assert.AreEqual(testResult.Pid, );
}
}
}

这是一个实际的集成测试用例代码,有一个当前测试类共用的服务地址,这个地址是DEV环境的,当然你也可以定义其他几个环境的服务地址,前提是环境是允许你连接的,那才有实际意义。

我们来看测试用例,它是一个查询方法测试用例,用来对ProductServiceClient.GetProductByPid服务方法进行测试,由于面向查询的操作是等幕的,不论我们查询多少次这个ID的Product,都不会对数据造成影响,但是如果我们测试的是一个更新或者删除就会带来问题。

在DEV环境中,测试更新、删除用例没有问题,但是如果你的机器是能够连接到远程某个生产或者PRD测试上时会带来一定的危险性,特别是在忙的时候,加班加点的干进度,你很难记住你当前的机器的host配置中是否还连接着远程的生产机器上,或者根本就不需要配置host就能够连接到某个你不应该连接的环境上。

这是目前的问题,那么我们如何解决这个问题呢 ,我们通过对测试代码进行一个简单的重构就可以避免由于连接到不该连接的环境中运行危险的测试用例。

其实很多时候,重构真的能够帮助我们找到出口,就好比俗话说的:"出口就在转角处“,只有不断重构才能够逐渐的保证项目的质量,而这种效果是很难得的。

提取抽象基类,对测试要访问的环境进行明确的定义。

 namespace OrderManager.Test
{
public abstract class ProductServiceIntegrationBase
{
/// <summary>
/// service address.
/// </summary>
protected const string ServiceAddressForDev = "http://dev.service.ProductService/"; /// <summary>
/// service address.
/// </summary>
protected const string ServiceAddressForPrd = "http://Prd.service.ProductService/"; /// <summary>
/// service address.
/// </summary>
protected const string ServiceAddressTest = "http://Test.service.ProductService/";
}
}

对具体的测试类消除重复代码,加入统一的构造方法。

 using System;
using Microsoft.VisualStudio.TestTools.UnitTesting; namespace OrderManager.Test
{
using ProductService.Contract; /// <summary>
/// Product service integration tests.
/// </summary>
[TestClass]
public class ProductServiceIntegrationTest : ProductServiceIntegrationBase
{
/// <summary>
/// product service client.
/// </summary>
private ProductServiceClient serviceInstance; /// <summary>
/// Initialization test instance.
/// </summary>
[TestInitialize]
public void InitTestInstance()
{
serviceInstance = ProductServiceClient.CreateClient(ServiceAddressForDev/*for dev*/);
} /// <summary>
/// Product service get product by pid test.
/// </summary>
[TestMethod]
public void ProductService_GetProductByPid_Test()
{
var testResult = serviceInstance.GetProductByPid(); Assert.AreNotEqual(testResult, null);
Assert.AreEqual(testResult.Pid, );
} /// <summary>
/// Product service delete search index test.
/// </summary>
[TestMethod]
public void ProductService_DeleteProductSearchIndex_Test()
{
var testResult = serviceInstance.DeleteProductSearchIndex(); Assert.IsTrue(testResult);
}
}
}

消除重复代码后,我们需要加入对具体测试用例检查是否能够连接到某个环境中去。我加入了一个DeleteProductSearchIndex测试用例,该用例是用来测试删除搜索索引的,这个测试用例只能够在本地DEV环境中运行(你可能觉得这个删除接口不应该放在这个服务里,这里只是举一个例子,无需纠结)。

为了能够有一个检查机制能提醒开发人员你目前连接的地址是哪一个,我们需要借助于测试上下文。

重构后,我们看一下现在的测试代码结构。

 using System;
using Microsoft.VisualStudio.TestTools.UnitTesting; namespace OrderManager.Test
{
using ProductService.Contract; /// <summary>
/// Product service integration tests.
/// </summary>
[TestClass]
public class ProductServiceIntegrationTest : ProductServiceIntegrationBase
{
/// <summary>
/// product service client.
/// </summary>
private ProductServiceClient serviceInstance; /// <summary>
/// Initialization test instance.
/// </summary>
[TestInitialize]
public void InitTestInstance()
{
serviceInstance = ProductServiceClient.CreateClient(ServiceAddressForPrd/*for dev*/); this.CheckCurrentTestCaseIsRun(this.serviceInstance);//check current test case .
} /// <summary>
/// Product service get product by pid test.
/// </summary>
[TestMethod]
public void ProductService_GetProductByPid_Test()
{
var testResult = serviceInstance.GetProductByPid(); Assert.AreNotEqual(testResult, null);
Assert.AreEqual(testResult.Pid, );
} /// <summary>
/// Product service delete search index test.
/// </summary>
[TestMethod]
public void ProductService_DeleteProductSearchIndex_Test()
{
var testResult = serviceInstance.DeleteProductSearchIndex(); Assert.IsTrue(testResult);
}
}
}

我们加入了一个很重要的测试实例运行时方法InitTestInstance,该方法会在测试用例每次实例化时先执行,在方法内部有一个用来检查当前测试用例运行的环境
this.CheckCurrentTestCaseIsRun(this.serviceInstance);//check current test case .,我们转到基类中。

 using System;
using Microsoft.VisualStudio.TestTools.UnitTesting; namespace OrderManager.Test
{
public abstract class ProductServiceIntegrationBase
{
/// <summary>
/// service address.
/// </summary>
protected const string ServiceAddressForDev = "http://dev.service.ProductService/"; /// <summary>
/// get service address.
/// </summary>
protected const string ServiceAddressForPrd = "http://Prd.service.ProductService/"; /// <summary>
/// service address.
/// </summary>
protected const string ServiceAddressTest = "http://Test.service.ProductService/"; /// <summary>
/// Test context .
/// </summary>
public TestContext TestContext { get; set; } /// <summary>
/// is check is run for current test case.
/// </summary>
protected void CheckCurrentTestCaseIsRun(ProductService.Contract.ProductServiceClient testObject)
{
if (testObject.ServiceAddress.Equals(ServiceAddressForPrd))// Prd 环境,需要小心检查
{
if (this.TestContext.TestName.Equals("ProductService_DeleteProductSearchIndex_Test"))
Assert.IsTrue(false, "当前测试用例连接的环境为PRD,请停止当前用例的运行。");
}
else if (testObject.ServiceAddress.Equals(ServiceAddressTest))//Test 环境,检查约定几个用例
{
if (this.TestContext.TestName.Equals("ProductService_DeleteProductSearchIndex_Test"))
Assert.IsTrue(false, "当前测试用例连接的环境为TEST,为了不破坏TEST环境,请停止用例的运行。");
}
}
}
}

在检查方法中我们使用简单的判断某个用例不能够在PRD、TEST环境下执行,虽然判断有点简单,但是在真实的项目中足够了,简单有时候是一种设计思想。我们运行所有的测试用例,查看各个状态。

一目了然,更为重要的是它不会影响你对其他用例的执行。当你在深夜12点排查问题的时候,你很难控制自己的眼花、体虚导致的用例执行错误带来的大问题,甚至是无法挽回的的错误。

此文献给那些跟我一样的.NET程序员们,通过简单的重构,我们解放了自己。

作者:王清培

出处:http://www.cnblogs.com/wangiqngpei557/

本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面

.NET程序员项目开发必知必会—Dev环境中的集成测试用例执行时上下文环境检查(实战)的更多相关文章

  1. 迈向高阶:优秀Android程序员必知必会的网络基础

    1.前言 网络通信一直是Android项目里比较重要的一个模块,Android开源项目上出现过很多优秀的网络框架,从一开始只是一些对HttpClient和HttpUrlConnection简易封装使用 ...

  2. Android程序员必知必会的网络通信传输层协议——UDP和TCP

    1.点评 互联网发展至今已经高度发达,而对于互联网应用(尤其即时通讯技术这一块)的开发者来说,网络编程是基础中的基础,只有更好地理解相关基础知识,对于应用层的开发才能做到游刃有余. 对于Android ...

  3. .NET零基础入门09:SQL必知必会

    一:前言 仿佛到了更进一步的时候了,每一个程序员迟早都会遇到数据存储的问题.我们拿什么来存储程序产生的数据?举例来说,用什么来存储我们的打老鼠游戏每次的成绩呢?选择如下: 1:内存中.缺点,退出游戏, ...

  4. 2015 前端[JS]工程师必知必会

    2015 前端[JS]工程师必知必会 本文摘自:http://zhuanlan.zhihu.com/FrontendMagazine/20002850 ,因为好东东西暂时没看懂,所以暂时保留下来,供以 ...

  5. [ 学习路线 ] 2015 前端(JS)工程师必知必会 (2)

    http://segmentfault.com/a/1190000002678515?utm_source=Weibo&utm_medium=shareLink&utm_campaig ...

  6. H5系列之History(必知必会)

    H5系列之History(必知必会)   目录 概念 兼容性 属性 方法 H5方法       概念     理解History Api的使用方式 目的是为了解决哪些问题   作用:ajax获取数据时 ...

  7. 学习《SQL必知必会(第4版)》中文PDF+英文PDF+代码++福达BenForta(作者)

    不管是数据分析还是Web程序开发,都会接触到数据库,SQL语法简洁,使用方式灵活,功能强大,已经成为当今程序员不可或缺的技能. 推荐学习<SQL必知必会(第4版)>,内容丰富,文字简洁明快 ...

  8. 读《MySQL必知必会》我学到了什么?

    前言 最近在写项目的时候发现自己的SQL基本功有些薄弱,遂上知乎查询MYSQL关键字,期望得到某些高赞答案的指点,于是乎发现了 https://www.zhihu.com/question/34840 ...

  9. 第4节:Java基础 - 必知必会(中)

    第4节:Java基础 - 必知必会(中) 本小节是Java基础篇章的第二小节,主要讲述抽象类与接口的区别,注解以及反射等知识点. 一.抽象类和接口有什么区别 抽象类和接口的主要区别可以总结如下: 抽象 ...

随机推荐

  1. [AngularJS] AngularJS系列(5) 中级篇之动画

    目录 CSS定义 JS定义 ng动画实际帮我们在状态切换的时候 添加特定的样式 从而实现动画效果. 一般我们会通过C3来实现具体的动画. CSS定义 ng-if 图(实际上,图并不能展现出什么): H ...

  2. 《Head First 设计模式》之装饰者模式

    作者:Grey 原文地址:http://www.cnblogs.com/greyzeng/p/5922248.html 模式名称 装饰者模式(Decorator Pattern) 需求 定义咖啡厅中的 ...

  3. JavaScript中的匿名函数及函数的闭包

    1.匿名函数 函数是JavaScript中最灵活的一种对象,这里只是讲解其匿名函数的用途.匿名函数:就是没有函数名的函数. 1.1 函数的定义,首先简单介绍一下函数的定义,大致可分为三种方式 第一种: ...

  4. .Net(c#)模拟Http请求之HttpWebRequest封装

    一.需求: 向某个服务发起请求获取数据,如:爬虫,采集. 二.步骤(HttpWebRequest): 无非在客户端Client(即程序)设置请求报文(如:Method,Content-Type,Age ...

  5. Extjs Panel

    刚学习Extjs @{ Layout = "~/_SiteLayout.cshtml"; Page.Title = "欢迎访问我的网站!"; } @{ stri ...

  6. visual studio 2015 开发android

    转载请注明: http://www.cnblogs.com/sunyl/p/5493249.html http://www.cnblogs.com/sunyl/ 最近有不少新闻, 甲骨文向谷歌索赔93 ...

  7. 疯狂Android讲义 - 学习笔记(四)

    Android应用通常有多个Activity,多个Activity组成Activity栈,当前活动的Activity位于栈顶.Activity相当于Java Web开发的Servlet.当Activi ...

  8. Netbeans 8.2关于PHP的新特性

    Netbeans 8.2在这个国庆期间终于发布了,其与PHP相关的新特性主要有: 支持PHP 7 详见前面翻译的一篇文章:Netbeans 8.2将支持PHP 7 编辑器功能增强 文档好像没有明确说明 ...

  9. border:0与border:none区别与联系

    联系:前台效果均实现了无边框 区别: 要解释区别,首先得先介绍一下border这个属性. border是一个简写属性.可以设置如下属性 border-width border-style border ...

  10. Java--缓存热点数据,最近最少使用算法

    1.最近最少使用算法LRU (Least recently used,最近最少使用) [实现]:最常见的是使用一个链表保存缓存数据 1.新数据插入到链表头部: 2.每当缓存命中(即缓存数据被访问),将 ...