翻看自己以前写的代码,突然间看到以前模拟的一个自动化测试架构的实现。幸好自己有写学习笔记的习惯,整理一下,贴出来,以备忘。
1. 利用反射机制,作为特性的元数据可以反过来在运行时影响代码的运行配置项,(例如:Windows Form程序中[STAThread]和[MTAThread用于表明应用程序的线程模型是单线程单元还是多线程--是否可以这样归属有待商榷)或者为特殊的操作方法以特性作标记,以便在运行时做特殊处理(在P/Invoke中DLLImport特性标记方法转化为托管方法)。
2. 可以用于构建管理项目程序集工具:特性表现为某种特殊注释,而注释内容可以在编译后从程序集(例如TestCaseClass.DLL)中读取出来,从而可以通过特性内容的注释和读取实现对程序集中各种类型和方法进行管理(例如可以筛选特定特性的方法执行)。
2. 特性参数分为环境参数和命名参数,如果特性类有自己的构造函数,则参数是必选的,为命名参数;剩下的参数为命名参数。在特性中应用命名参数必须加上参数名称。可以重载构造函数以满足不同组合。
namespace AttributesClass { [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)] public sealed class ClassCleanupAttribute:Attribute { public ClassCleanupAttribute() { } } } namespace AttributesClass { [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)] public sealed class ClassInitializeAttribute:Attribute { public ClassInitializeAttribute() { } } } namespace AttributesClass { [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)] public sealed class DescriptionAttribute:Attribute { private string description; public string Description { get { return this.description; } set { this.description = value; } } public DescriptionAttribute(string text) { this.description = text; } } } namespace AttributesClass { [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)] public sealed class PriorityAttribute:Attribute { private int priorityLevel; public int Level { get { return this.priorityLevel; } set { this.priorityLevel = value; } } public PriorityAttribute(int level) { this.priorityLevel = level; } } } namespace AttributesClass { [AttributeUsage(AttributeTargets.Method, AllowMultiple = true)] public sealed class TestPropertyAttribute:Attribute { #region Fileds private string propertyName = null; private string propertyValue = null; #endregion #region Proerties public string Name { get { return this.propertyName; } set { this.propertyName = value; } } public string Value { get { return this.propertyValue; } set { this.propertyValue = value;} } #endregion #region Constructors public TestPropertyAttribute(string strPropertyName, string strPropertyValue) { this.propertyName = strPropertyName; this.propertyValue = strPropertyValue; } #endregion } }
查找TestProperty限定的方法,找到后查找所在类是否有ClassInitialize和ClassCleanup方法,如果有则先执行前者然后执行该方法最后执行后者。
public void RunTest(string name, string value) { List<string> selectedMethods = SearchMethodByTestProperty(name, value); foreach (string method in selectedMethods) { string[] strSplitMethod = method.Split('.'); string nameSpaceName = strSplitMethod[0]; string className = strSplitMethod[1]; string methodName = strSplitMethod[2]; Console.WriteLine(nameSpaceName + '.' + className); Console.WriteLine("-------------------------------------------------------"); Type typeClass = assmeblyObject.GetType(nameSpaceName + '.' + className); MethodInfo testMethod = typeClass.GetMethod(methodName); object classObject = assmeblyObject.CreateInstance(nameSpaceName+'.'+className); MethodInfo[] methodArray = typeClass.GetMethods(); foreach (MethodInfo objMethod in methodArray) { object[] classInitializeAttributes = objMethod.GetCustomAttributes(typeof(AttributesClass.ClassInitializeAttribute), true); if (classInitializeAttributes.Length == 1) { Console.WriteLine("You are invoking the classInitialize method..."); Console.WriteLine(objMethod.Name + @"() "); objMethod.Invoke(classObject, null); //InvokeIntialize(classObject, objMethod); } } //InvokeTestMethod(classObject, method); Console.WriteLine("You are invoking the Select method..."); Console.WriteLine(methodName + @"() "); testMethod.Invoke(classObject, null); foreach (MethodInfo objMethod in methodArray) { object[] classCleanupAttributes = objMethod.GetCustomAttributes(typeof(AttributesClass.ClassCleanupAttribute), true); if (classCleanupAttributes.Length == 1) { //InvokeCleanup(classObject, objMethod); Console.WriteLine("You are invoking the classCleanup method..."); Console.WriteLine(objMethod.Name + @"() "); objMethod.Invoke(classObject, null); } } Console.WriteLine(""); } }
模拟实现架构代码笔记:
1. 特性定义应该为public Sealed 以避免继承和便于命名空间外的类或函数调用(自定义特性通常情况下很少被继承,并且定义好的特性大多数情况下还是命名空间外引用的)
2. 定义特性时应该使用[AttributeUsage(AttributeTargets.Method, AllowMultiple = true)]来指定该特性是用于标记方法,类还是成员变量特性的。
3. 别调用的测试类DemoTest由于会被主调用函数调用也应该声明为public。
4. 约定测试程序集里所有方法种类一致便于管理.(倘若测试Assmbly里既有静态方法又有非静态方法的话,那么在主调用函数反射时必须考虑到不同种类的方法,invoke 时的不同,非静态方法必须得先实例化才能invoke)
5. 该架构里所有方法都是非静态的,主调用函数实现非静态方法必须得先实例化一个类,显然比较麻烦。 为什么不写成非静态的呢。(因为倘若用到某个类中的方法,必须默认得去调用Initialize和Cleanup函数[如果这些函数存在的话],如果用静态方法的话就必须的自己去判定查找调用这些方法,显然实现起来比较麻烦)
不足:
1 并没有完成对多限定条件下的查询方法
2 目前为止所有的测试方法必须不包含参数(包含参数的方法很容易实现,可以考虑在运行时XML传递参数)
3 PropertyDemoTest类结构不太清晰,可以考虑重构一下
4 可以增加更多的自定义特性完成更为复杂的操作
由于只是简单模拟框架,实现功能较为简单,更为复杂的功能可以以后再加,这个架构最大的好处是耦合度小,可扩展性强。
全部源码可去csdn下载,连接:http://download.csdn.net/detail/fl815824/4340631
http://msdn.microsoft.com/zh-cn/library/f7ykdhsy(v=VS.80).aspx