C#基础系列——Attribute特性使⽤
前⾔:上篇总结了下反射得基础⽤法,这章我们来看看C#的另⼀个基础技术——特性。
1、什么是特性:就博主的理解,特性就是在类的类名称、属性、⽅法等上⾯加⼀个标记,使这些类、属性、⽅法等具有某些统⼀的特征,从⽽达到某些特殊的需要。⽐如:⽅法的异常捕捉,你是否还在某些可能出现异常的地⽅(例如数据库的操作、⽂件的操作等)经常使⽤atch。这个时候如果使⽤特性,就可以⼤⼤减少⽅法⾥⾯的atch的使⽤。你只需要定义⼀个专门捕捉异常的特性类ExceptionExAttribute,然后给这个特性类做些特殊处理,⽐如给它增加⼀个AOP拦截的功能(AOP拦截的⽅式很多,有兴趣可以搜搜看,园⼦⾥⾯很多类似的⽂章)。那么在可能出现异常的⽅法名称上⾯加上⼀个[ExceptionEx]特性标签,这个⽅法就具有⾃动捕捉异常的能⼒。还是加上官⽅定义:
特性提供功能强⼤的⽅法,⽤以将元数据或声明信息与代码(程序集、类型、⽅法、属性等)相关联。特性与程序实体关联后,即可在运⾏时使⽤名为“反射”的技术查询特性。
特性具有以下属性:
特性可向程序中添加元数据。元数据是有关在程序中定义的类型的信息。所有的 .NET 程序集都包含指定的⼀组元数据,这些元数据描述在程序集中定义的类型和类型成员。可以添加⾃定义特性,以指定所需的任何附加信息。
可以将⼀个或多个特性应⽤到整个程序集、模块或较⼩的程序元素(如类和属性)。
特性可以与⽅法和属性相同的⽅式接受参数。
程序可以使⽤反射检查⾃⼰的元数据或其他程序内的元数据。
(以上来⾃MSDN)
2、为什么需要特性:这个上⾯已经简单介绍过,特性能⼤⼤减少统⼀需求的代码量。其他不说,⾄少它能让我们的代码看上去更⼤⽓点吧~~
3、特性的使⽤:博主这次还是打算从三个⽅便分别介绍下特性的常规使⽤⽅法。当然这⼏种⽅式都是博主原来⽤过的,可能不是最好的举例场景,但是也算⽐较典型的特性⽤法吧。
(1)类的属性上⾯特性的⽤法:
之所以将这个放在最前⾯介绍是因为博主最近做的⼀个BS项⽬正好⽤到,并且使⽤场景也⽐较典型。⾸先介绍下使⽤场景:最近项⽬有⼀个需求,BS界⾯需要⼀个拖拽的功能。如下图
writeline特点
当将左边的3个div拖到右边来时,每个div都有⾃⼰的特有属性,⽐如2部门拖过来时,要显⽰如下属性:
1和3部门拖过来时可能对应的属性不同。
设计思路:每个div对应的Model,每个Model⾥⾯有⾃⼰特有的属性,然后属性上⾯加上特性显⽰属
性的名称和默认值,以及界⾯应该呈现的html标签。
实现代码:
⾸先来看⾃定义的⼀个特性类:
  public class DetailAttribute : Attribute
{
public string AttrName { set; get; }
public string Html { set; get; }
public string DefaultValue { set; get; }
public string DataSource { set; get; }
}
对应的Model:
  public class Factory
{
[Detail(AttrName="宽度", Html="<input type='text' />", DefaultValue="50", DataSource=null)]
public string Width { set; get; }
[Detail(AttrName = "⾼度", Html = "<input type='text' />", DefaultValue = "50", DataSource = null)]
public string Height { set; get; }
[Detail(AttrName = "状态", Html = "<select></select>", DefaultValue = null, DataSource = "select text,value from status")]
public string Status { set; get; }
[Detail(AttrName = "Tag值", Html = "<input type='text' />", DefaultValue = "", DataSource = null)]
public string Tag { set; get; }
}
public class FactoryDetail
{
[Detail(AttrName = "宽度", Html = "<input type='text' />", DefaultValue = "50", DataSource = null)]
public string Width { set; get; }
[Detail(AttrName = "⾼度", Html = "<input type='text' />", DefaultValue = "50", DataSource = null)]
public string Height { set; get; }
[Detail(AttrName = "状态", Html = "<select></select>", DefaultValue = null, DataSource = "select text,value from status")]
public string Status { set; get; }
[Detail(AttrName = "Tag值", Html = "<input type='text' />", DefaultValue = "", DataSource = null)]
public string Tag { set; get; }
[Detail(AttrName = "描述", Html = "<input type='text' />", DefaultValue = "", DataSource = null)]
public string Desc { set; get; }
}
然后在界⾯的拖放事件结束时通过js发送ajax请求来得到界⾯要呈现的html:
$(".jq-draggable-outcontainer").draggable({
helper: "clone",
scroll: true,
drag: function (event, ui) {
//  debugger;
}
});
$("#content").droppable({
drop: function (event, ui) {
//  debugger;
if (ui.draggable[0].className.indexOf("jq-draggable-outcontainer") > 0) {
var text = ui.draggable[0].im();
$(this).append('<div class="window jq-draggable-incontainer" onclick="GetPropertiesByType(\'1\',this)" id="                $("#content2").html("");
cur_selector = $("#window"+iIndex);
$.Ewin.AjaxPost("/Home/GetModelByType", { strType: "Factory" }, function (data, status) {
var element = $.parseJSON(data.Json);
var arrProp = element.element.property;
//0.构造html
var strHtml = "<div style='float:right;padding-top:0px;width:300px;height:auto;'><table cellpadding='5' border='1'>";
//1.拼html构造属性
strHtml += "</table></div>";
$("#content2").append(strHtml);
}, function () {
}, null);
iIndex++;
}
}
});
对应的C#⽅法:
    public JsonResult GetModelByType(string strType)
{
       //strType传过来的是Factory或者FactoryDetail
var assembly = Assembly.Load("Ewin.Client.Web");//参数为程序集的名称
var oType = assembly.GetType("Ewin.Client.Web.Controllers." + strType);
        //得到类的所有属性
var lstProperties = oType.GetProperties();
foreach (var oProperty in lstProperties)
{
          //得到每⼀个属性的特性类集合
IList<CustomAttributeData> lstAttr = oProperty.GetCustomAttributesData();
foreach (var oAttr in lstAttr)
{
            //得到每⼀个特性类的全称
Console.WriteLine("特性类的名称" + oAttr.AttributeType.FullName);
Console.WriteLine("特性类成员如下:");
            //得到特性类的所有参数
var lstAttrArgu = oAttr.NamedArguments;
foreach (var oAttrAru in lstAttrArgu)
{
              //取每个特性类参数的键值对
Console.WriteLine(oAttrAru.MemberName + "=" + oAttrAru.TypedValue.Value);
}
//Console.WriteLine(oAttr.AttributeType+"——"+oAttr.NamedArguments);
}
}
return Json(new { }, JsonRequestBehavior.AllowGet);
}
GetModelByType⽅法结果简单构造下然后将属性的键值对返回给js⽅法,然后再由js追加到界⾯上⾯。这样通过特性和反射的结合能很快完成这个⼩功能的设计。
(2)类的⽅法上⾯特性的⽤法:
这个⽤法.Net framework⾥⾯就很多,如果MVC⾥⾯Filter过滤器的⽤法:
public class SuperLogStat : ActionFilterAttribute
{
//模块名称
private EnumModuleName moduleEnum = EnumModuleName.ModuleOther;
//功能名称
private string functionName = string.Empty;
//⽤户Id
private string userId = string.Empty;
public string Version
{
get { return ConfigurationManager.AppSettings["UploatStatVersion"]; }
}
public EnumModuleName ModuleEnum
{
get { duleEnum; }
set { duleEnum = value; }
}
public string FunctionName
{
get { return this.functionName; }
set { this.functionName = value; }
}
  //这两个⽅法都是⽗类的virtual⽅法,⼀个再return View()之前执⾏,⼀个再之后执⾏
//
/
/ 摘要:
//    在执⾏操作⽅法之前由 MVC 框架调⽤。
//
// 参数:
//  filterContext:
//    筛选器上下⽂。
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
try
{
string userName = filterContext.HttpContext.User.Identity.Name;
this.userId = userName.Replace("china\\", "");
}
catch (Exception)
{
this.userId = string.Empty;
}
}
//
// 摘要:
//    在执⾏操作结果之前由 MVC 框架调⽤。
//
/
/ 参数:
//  filterContext:
//    筛选器上下⽂。
public override void OnResultExecuting(ResultExecutingContext filterContext)
{
UserLogUtils.LogUserStatic(this.userId, this.Version,
}
}
View Code
在Controller⾥⾯⽅法上⾯加上特性:
//调⽤
[SuperLogStat(ModuleEnum = EnumModuleName.ModuleHome, FunctionName = "待审核")] public ActionResult MyApplyToAuditing()
{
return View();
}
这个ActionFilterAttribute这个特性⽤法⾥⾯就有异常的拦截机制,和前⾯说的⾃定义的异常拦截是相同的。
(3)类上⾯特性的⽤法:
类上⾯特性的⽤法其实.Net⾥⾯也很多。⽐如为了避免new⼀个对象⽽使⽤的MEF就是⼀个很有说服⼒的例⼦:在定义个类实现⼀个接⼝时:
[Export("Impc_TB_Test", typeof(Ifc_TB_Test))]
public class Impc_TB_Test : Ifc_TB_Test
{
......
}
定义接⼝没有任何特殊:
public interface Ifc_TB_Test
{
......
}
然后在使⽤时只需要加⼀个[Import]标签,这个变量就会在编译时⾃动new⼀个Impc_TB_Test变量:[Import("Impc_TB_Test")]
Ifc_TB_Test service { get; set; }
在使⽤service变量时,就可以直接把它当做⼀个Impc_TB_Test对象来使⽤。是不是很⽅便。
这⼏种常见⽤法都是博主⽤过的觉得⽐较好的场景,当然特性的⽤法肯定远不⽌如此,欢迎⼤侠们指正拍砖~~

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。