这篇文章两个目的,一是开阔设计的思路,二是实例代码可以拿来就用。

设计的思路来源于《Effective c#》第一版Item 24: 优先使用声明式编程而不是命令式编程。特别的地方是,希望提供多个属性的默认排序,而不仅仅只根据一个属性,另外一点是,优先调用对象属性实现了的IComparable<T>接口,如果没有实现接口,才调用IComparable进行比较。排序类实现泛型,得到类型安全。

总的思路:Attribute用来装饰我们想要获取元数据的类,使用Reflection来提取元数据,根据提取到的元数据实现一些和对象无关的组件功能。

那么,这个例子要实现的效果是用Attribute装饰类对象,设置该对象的默认排序属性,排序的时候,根据这些默认排序来进行排序。

 [DefaultSort(new string[] {"ID", "Name"})]
class SortData
{
public int ID { get; set; } public string Name { get; set; } public string Value { get; set; } public override string ToString()
{
return String.Format("ID:{0},Name:{1},Value:{2}", ID, Name, Value);
}
}

对于SortData对象来说,我们希望根据它的ID来排序,如果ID相等,再根据Name属性来排序。像它的名字暗示的一样,这是默认的行为,不需要我们实现SortData的IComparable<SortData>接口,将来要改变排序规则,只要修改DefaultSort中属性名称数组的内容就够了,很方便。

原书中记录的DefaultAttribute只能根据一个属性名称来排序,不够实用,希望它像下面的类一样,能记录多个属性的名称。

 [AttributeUsage(AttributeTargets.Class | AttributeTargets.Struct, AllowMultiple=false)]
public class DefaultSortAttribute : System.Attribute
{
private string[] m_propNames; public string[] PropNames
{
get { return m_propNames; }
set { m_propNames = value; }
} public DefaultSortAttribute(string propName)
{
m_propNames = new string[] { propName };
} public DefaultSortAttribute(string[] propNames)
{
m_propNames = propNames;
}
}

注意仍然保留了只希望拿一个属性来排序的构造函数,对类进行装饰时,往类上面放[DefaultSort(new string[] {"ID", "Name"})] 和[DefaultSort("ID")]类似的声明就够了。

既然使用Attribute装饰了类,就要知道这样的元数据,下面需要采用Reflection读到要排序的默认属性名,相对于原书中的改进是,使用泛型和优先使用属性的IComparable<T>接口来比较排序。

 using System;
using System.Linq;
using System.Collections;
using System.Collections.Generic;
using System.ComponentModel;
using System.Reflection; namespace ProJKY.Extensions
{
public class DefaultSortComparer<T> : IComparer, IComparer<T>
{
private readonly PropertyDescriptor[] m_sortProps;
private readonly bool m_reverse = false;
private readonly bool m_valueType = false; public DefaultSortComparer() :
this(false)
{ } public DefaultSortComparer(bool reverse)
{
m_reverse = reverse;
Type t = typeof(T);
m_valueType = t.IsValueType; object[] a = t.GetCustomAttributes(typeof(DefaultSortAttribute), false); // 强制检查,不支持没有用DefaultSortAttribute装饰的类
if (a.Length != )
throw new NotSupportedException(t.Name); DefaultSortAttribute sortName = a[] as DefaultSortAttribute; string[] propNames = sortName.PropNames; m_sortProps = new PropertyDescriptor[propNames.Length]; PropertyDescriptorCollection props = TypeDescriptor.GetProperties(t); for (int i = ; i < propNames.Length; i++){
foreach (PropertyDescriptor p in props){
if (p.Name == propNames[i]){
m_sortProps[i] = p;
break;
}
}
}
} int IComparer.Compare(object left, object right)
{
if (HasNull(left, right) == true)
{
int nullCompare = CompareWithNull(left, right); return m_reverse ? -nullCompare : nullCompare;
} if (left.GetType() != right.GetType())
throw new ArgumentException("left and right not match."); if (typeof(T).IsAssignableFrom(left.GetType()) == false)
throw new ArgumentException("type not compatible."); return Compare((T)left, (T)right);
} public int Compare(T x, T y)
{
if (m_valueType == false && HasNull(x, y) == true){
int nullCompare = CompareWithNull(x, y);
return m_reverse ? -nullCompare : nullCompare;
} foreach (var prop in m_sortProps){
object xValue = prop.GetValue(x);
object yValue = prop.GetValue(y); if (HasNull(xValue, yValue) == true){
int nullCompare = CompareWithNull(xValue, yValue); return m_reverse ? -nullCompare : nullCompare;
} Type propType = xValue.GetType(); // 优先使用IComaprable<T>接口
if (typeof(IComparable<>).MakeGenericType(propType).IsAssignableFrom(propType))
{
MethodInfo methodInfo = propType.GetMethods().FirstOrDefault(method => method.Name == "CompareTo"
&& method.GetParameters().Length ==
&& method.GetParameters()[].ParameterType == propType); int gretValue = (int)methodInfo.Invoke(xValue, new object[] { yValue }); if (gretValue == ) continue; return m_reverse ? -gretValue : gretValue;
} IComparable xNonGeneric = xValue as IComparable;
IComparable yNonGeneric = yValue as IComparable; if (xNonGeneric == null)
throw new ArgumentException("Property " + prop.Name + " is not comparable."); int retValue = xNonGeneric.CompareTo(yValue); if (retValue == ) continue; return m_reverse ? -retValue : retValue;
} return ;
} int CompareWithNull(object left, object right)
{
if ((left == null) && (right == null))
return ; if (left == null)
return -; return ;
} bool HasNull(object left, object right)
{
if (left == null || right == null)
return true;
return false;
}
}
}

需要注意的是DefaultSortComparer<T>是泛型的,并实现了IComparer, IComparer<T>接口,实现了这两个接口,才方便排序。

代码写贴了这么多,用起来怎么用呢。

 var data1 = new SortData() { ID = , Name = "info", Value = "key"};
var data2 = new SortData() { ID = , Name = "64File", Value = "license" };
var data3 = new SortData() { ID = , Name = "cloneToken", Value = "comparer" };
var data4 = new SortData() { ID = , Name = "0est", Value = "backend" }; List<SortData> sortData = new List<SortData>();
sortData.Add(data1);
sortData.Add(data2);
sortData.Add(data3);
sortData.Add(data4); sortData.Sort(new DefaultSortComparer<SortData>(false)); sortData.ForEach(data => Console.WriteLine(data));

结果就不献丑了,经测试,能正常工作。

通过这个例子,就可以看到,要实现它的关键是,Attribute负责装饰类,Reflection负责读取特定Attribute装饰后的元数据信息,实现和特定类类型无关的组件。一次Coding,多次复用。

希望大家多支持,以后会多放一些有意思的开箱即用的代码上来。

Attribute+Reflection,提高代码重用的更多相关文章

  1. MVC引入SERVICE层 提高代码重用性 沟通CONTROL和MODEL

    MVC是web开发中常见的程序结构. 简单的mvc结构如下: view层:显示层. control层:业务层,集合了各种action. model层:模型层,一般和数据打交道.简单的sample:一个 ...

  2. Android 命名规范 (提高代码可以读性)

    android文件众多,根据名称来辨别用途很重要,因此命名要规范 这篇文章可参考:Android 命名规范 (提高代码可以读性) 刚接触android的时候,命名都是按照拼音来,所以有的时候想看懂命名 ...

  3. 精简CSS代码,提高代码的可读性和加载速度

    前言 提高网站整体加载速度的一个重要手段就是提高代码文件的网络传输速度.之前提到过,所有的代码文件都应该是经过压缩了的,这可提高网络传输速度,提高性能.除了压缩代码之外,精简代码也是一种减小代码文件大 ...

  4. “前.NET Core时代”如何实现跨平台代码重用 ——程序集重用

    除了在源代码层面实现共享("前.NET Core时代"如何实现跨平台代码重用 --源文件重用)之外,我们还可以跨平台共享同一个程序集,这种独立于具体平台的"中性" ...

  5. PHP代码重用与函数编写

    代码重用与函数编写 1.使用require()和include()函数 这两个函数的作用是将一个文件爱你载入到PHP脚本中,这样就可以直接调用这个文件中的方法.require()和include()几 ...

  6. 提高代码质量 CheckStyle FindBugs PMD

    提高代码质量-工具篇 注:这是一篇翻译文章,原文:How to improve quality and syntax of your Android code,为了理解连贯,翻译过程中我修改了一些陈述 ...

  7. 谈谈Delphi中的类和对象4---类是一种对数据和操作高度的封装机制 && 类是一种代码重用机制

    五.类是一种对数据和操作高度的封装机制 1)数据封装 unit Unit2; interface type TEmployee = class; private FName: String; publ ...

  8. 从零开始攻略PHP(6)——代码重用与函数编写的一些注意事项

    一个新的项目是这样创建的:它将已有的可重新利用的组件进行组合,并将新的开发难度降低到最小. 代码重用的好处:降低成本.提升可靠性和一致性. 1.使用require()和include()函数 使用一条 ...

  9. 利用Apply的参数数组化来提高代码的优雅性,及高效性

    利用Apply的参数数组化来提高代码的优雅性,及高效性 Function.apply()在提升程序性能方面的技巧 我们先从Math.max()函数说起,Math.max后面可以接任意个参数,最后返回所 ...

随机推荐

  1. java使用freemarker生成word文档

    1.原料 开源jar包freemarker.eclipse.一份模板word文档 2.首先设计模板word文档 一般,通过程序输出的word文档的格式是固定的,例如建立一个表格,将表格的标题写好,表格 ...

  2. 本地jar包 安装到本地仓库中的命令

    maven 项目 本地jar包 安装到本地仓库中去: 首先进入到该文件所在文件夹内 若不在直接绝对路径就可以.注意命令中的空格 mvn install:install-file  -Dfile=文件名 ...

  3. WCF系列教程之WCF服务配置

    文本参考自:http://www.cnblogs.com/wangweimutou/p/4365260.html 简介:WCF作为分布式开发的基础框架,在定义服务以及消费服务的客户端时可以通过配置文件 ...

  4. solr实时更新mysql数据的方法

    第一步:创建core core是solr的特有概念,每个core是一个查询数据,.索引等的集合体,你可以把它想象成一个独立数据库,我们创建一个新core:名字[core1] 进入linux命令行,进入 ...

  5. ERROR 1064 (42000): You have an error in your SQL syntax;

    出现: ERROR 1064 (42000): You have an error in your SQL syntax; 1.SQL语句拼写错误. 具体很简单.慢慢查看 2.使用到了SQL关键字. ...

  6. JS中typeof和instanceof的用法和区别

    typeof和instanceof的区别 instanceof 常用来判断一个变量是否为空,或者是什么类型的.但它们之间还是有区别的 instanceof的用法 instanceof返回的是一个布尔值 ...

  7. Swagger与SpringMVC项目整合

    Swagger与SpringMVC项目整合 来源:http://www.2cto.com/kf/201502/376959.html 为了方便的管理项目中API接口,在网上找了好多关于API接口管理的 ...

  8. weblogic 异常常见处理方法

    1.系统日志 通过分析weblogic的系统日志及应用程序出错的地方,可能找到异常的原因. 2. 数据库连接 如果用户访问系统过慢,且连接池已经占满而weblogic的线程数量很少,就要查看应用是否没 ...

  9. Node.js事件驱动模型

    一.传统线程网络模型 在了解Node.js事件驱动模型之前,我们先了解一下传统的线程网络模型,请求进入web服务器(IIS.Apache)之后,会在线程池中分配一个线程来线性同步完成请求处理,直到请求 ...

  10. [转]C#进阶系列——WebApi 接口返回值不困惑:返回值类型详解

    本文转自:http://www.cnblogs.com/landeanfen/p/5501487.html 阅读目录 一.void无返回值 二.IHttpActionResult 1.Json(T c ...