C# 特性(Attribute)超详细教程

文章目录

  • 0.前篇
  • 1.特性概念
  • 2.特性的声明和使用
    • 2.1 特性定义语法
    • 2.2 特性目标
  • 3.预定义特性
    • 3.1 AttributeUsage
    • 3.2 Conditional
    • 3.3 其它预定义特性
  • 4.MyAttributeHelper(特性使用帮助类)
  • 5.特性应用
    • 5.1 添加说明信息并获取
    • 5.2 数据验证

0.前篇

学习本文前建议先阅读我的关于反射的博文:
https://editor.csdn.net/md/?articleId=139095147

1.特性概念

特性是用于在运行时传递程序中各种元素(比如类、方法、结构、枚举、组件等)的行为信息的声明性标签。

2.特性的声明和使用

2.1 特性定义语法

特性在声明时以Attribute结尾,在使用时可省去Attribute

[attribute(positional_parameters, name_parameter = value,)]
element

参数说明:

  • positional_parameters: 表示特性必须具备的信息。
  • name_paramete:表示特性可选的信息。
  • element: 在这里表示特性目标。

代码示例:
特性声明:

public class TestAttribute : Attribute
{public int Parm { get; set; }private int id;private string name;public TestAttribute(){}public TestAttribute(int id, string name){this.id = id;this.name = name;}
}

特性使用:

[Test]  // 使用无参构造函数
public class TestClass1
{}[Test(Parm = 123)]  // 使用无参构造函数 + 指定参数
public class TestClass2
{}[Test(1, "test")]   // 使用有参构造函数
public class TestClass3
{}

2.2 特性目标

特性目标指的是应用特性的实体。例如,特性目标可以是类、特定方法或整个程序集。一般情况,特性应用于紧跟在它后面的元素。不过,C# 特性支持显示标识,例如可以显示标识为将特性应用于方法,或者是应用于其参数或返回值。

显示标识特性目标的语法如下:

[target : attribute-list]
  • target:表示指定的特性目标值。
  • attribute-list:表示要应用的特性列表。

下表展示常用的 target 值:

目标值适用对象
assembly整个程序集
module当前程序集模块
field类或结构中的字段
event事件
method方法或 get 和 set 属性访问器
param方法参数或 set 属性访问器参数
propertyProperty(属性)
return方法、属性索引器或 get 属性访问器的返回值
type结构、类、接口、枚举或委托

代码示例:

// 默认: 应用于方法
[Test]
int Method1()
{ return 0; 
}// 显示指定应用于方法
[method: Test]
int Method2()
{ return 0; 
}// 应用于参数
int Method3([Test] string contract) 
{ return 0; 
}// 应用于返回值
[return: Test]
int Method4()
{ return 0; 
}

3.预定义特性

3.1 AttributeUsage

特性 AttributeUsage 描述了如何使用一个自定义特性类。注意,使用特性修饰的类 AttributeUsage 必须是 System.Attribute 的直接或间接派生类,否则将发生编译时错误

AttributeUsage 特性的语法如下:

[AttributeUsage(validon, AllowMultiple = allowmultiple, Inherited = inherited)]
  • validon: 表示可被应用的特性目标值。它是枚举器 AttributeTargets 的值的组合。默认值是 AttributeTargets.All。
  • AllowMultiple: 可选,allowmultiple 选项为其提供一个布尔值。表示特性是否能被重复放置多次
  • Inherited:可选,inherited 选项为其提供一个布尔值。表示 能否被派生类所继承。

示例代码:

[AttributeUsage(AttributeTargets.Class, AllowMultiple = false, Inherited = false)]
public class ExampleAttribute : Attribute
{public ExampleAttribute(string name){}
}

3.2 Conditional

特性 Conditional 是有条件的意思。使用特性修饰的方法也就是条件方法,条件方法的执行依赖于指定的预处理标识符。预处理标识符影响着方法调用的条件编译,这种影响取决于指定的值。例如事先用预处理器指令定义了一个 Test 字符,当条件方法指定的值也为 Test 时, 则该方法会被执行。

语法:

[Conditional(conditionalSymbol)]
  • 参数 conditionalSymbol 表示指定的预处理标识符。
#define Test
using System;public class Example
{[Conditional("Test")]static void Method1(){Console.WriteLine("Method1");}static void Method2(){Console.WriteLine("Method2");}static void Main(){Method1();Method2();}
}

输出:

Method1
Method2

注释掉 #define Test 输出

Method2

3.3 其它预定义特性

特性说明
[Obsolete]标记已过时的代码,使得在使用过时成员时发出警告或错误信息。
[Serializable]用于标记类,表示该类可以序列化,即可以在网络上传输或者在文件中存储。
[DllImport]用于指示要在程序中调用非托管代码(通常是 DLL)的方法。
[Conditional]与预处理指令 ‘#if’ 和 ‘#endif’ 结合使用,根据定义的条件编译代码。
[AttributeUsage]用于指定自定义特性的使用方式,如允许的目标类型和是否允许多次应用等。
[Conditional]根据条件编译代码,类似于预处理指令,但是使用 Attribute。
[DefaultValue]为属性或字段设置默认值。
[Description]为属性或者事件提供一个描述,通常在设计时使用。

4.MyAttributeHelper(特性使用帮助类)

using System.Reflection;namespace Ming.Utils
{public static class MyAttributeHelper{/// <summary>/// 获取该类型下所有的带Attribute的方法/// </summary>/// <typeparam name="T">特性类型</typeparam>/// <param name="type"></param>/// <returns></returns>public static List<MethodInfo> GetAllMethods<T>(Type type) where T : class, new(){var res = new List<MethodInfo>();res = type.GetMethods().Where(t => t.GetCustomAttributes(typeof(T), false).Any()).ToList();return res;}/// <summary>/// 获取该类型下所有的带Attribute的属性/// </summary>/// <typeparam name="T"></typeparam>/// <param name="type"></param>/// <returns></returns>public static List<PropertyInfo> GetAllPropertys<T>(Type type) where T : class, new(){var res = new List<PropertyInfo>();res = type.GetProperties().Where(t => t.GetCustomAttributes(typeof(T), false).Any()).ToList();return res;}/// <summary>/// 获取程序集所有带 T 特性的类class/// </summary>/// <typeparam name="T">特性类型</typeparam>/// <returns>程序集下所有带 T 特性的类class</returns>public static List<Type> GetAllTypes<T>() where T : Attribute{var res = new List<Type>();//Assembly存放所有的程序集res = Assembly.GetExecutingAssembly().GetTypes().Where(t => t.GetCustomAttributes(typeof(T), false).Any())//我们找到所有程序集中带有T特性的Type类型.ToList();return res;}/// <summary>/// 获取类上的特性/// </summary>/// <typeparam name="T"></typeparam>/// <param name="model"></param>/// <returns></returns>public static T GetAttribute<T>(Type type) where T : Attribute, new(){var res = new T();res = type.GetCustomAttribute<T>();return res;}/// <summary>/// 获取方法上的特性/// </summary>/// <typeparam name="T"></typeparam>/// <param name="model"></param>/// <returns></returns>public static T GetAttribute<T>(MethodInfo type) where T : Attribute, new(){var res = new T();res = type.GetCustomAttribute<T>();return res;}/// <summary>/// 获取属性上的特性/// </summary>/// <typeparam name="T"></typeparam>/// <param name="model"></param>/// <returns></returns>public static T GetAttribute<T>(PropertyInfo type) where T : Attribute, new(){var res = new T();res = type.GetCustomAttribute<T>();return res;}/// <summary>/// 返回带有Attribute的类型元祖列表/// </summary>/// <typeparam name="Att"></typeparam>/// <returns></returns>public static List<(Type type, Att att)> GetAll_TypeAndAtt<Att>() where Att : Attribute, new(){var res = new List<(Type type, Att att)> ();var typeLists = GetAllTypes<Att>();foreach (var item in typeLists){var att = GetAttribute<Att>(item);res.Add((item, att));   }return res;}/// <summary>/// 返回带有Attribute的变量元祖列表/// </summary>/// <typeparam name="Att"></typeparam>/// <param name="type"></param>/// <returns></returns>public static List<(PropertyInfo property, Att att)> GetAll_PropertyAndAtt<Att>(Type type) where Att : Attribute, new(){var res = new List<(PropertyInfo type, Att att)>();var typeLists = GetAllPropertys<Att>(type);foreach (var item in typeLists){var att = GetAttribute<Att>(item);res.Add((item, att));}return res;}/// <summary>/// 返回带有Attribute的方法元祖列表/// </summary>/// <typeparam name="Att"></typeparam>/// <param name="type"></param>/// <returns></returns>public static List<(MethodInfo method, Att att)> GetAll_MethodAndAtt<Att>(Type type) where Att : Attribute, new(){var res = new List<(MethodInfo type, Att att)>();var typeLists = GetAllMethods<Att>(type);foreach (var item in typeLists){var att = GetAttribute<Att>(item);res.Add((item, att));}return res;}}
}

5.特性应用

5.1 添加说明信息并获取

/// <summary>
/// 备注特性
/// </summary>
public class RemarkAttribute : Attribute
{private string Remark { get; set; }public RemarkAttribute(string Remark){this.Remark = Remark;}public string GetRemark(){return this.Remark;}
}// 枚举
public enum ESex
{[Remark("男")]male = 1,[Remark("女")]female = 2,
}/// <summary>
/// Enum扩展方法
/// </summary>
public static class EnumExtension
{public static string GetRemark(this Enum model){if (model is ESex){Type type = typeof(ESex);FieldInfo fi = type.GetField(model.ToString());object[] attributes = fi.GetCustomAttributes(true);foreach (var attr in attributes){if (attr is RemarkAttribute){RemarkAttribute remark = (RemarkAttribute)attr;return remark.GetRemark();}}}return string.Empty;}
}

使用:

Console.WriteLine(ESex.male.GetRemark());
// 男

5.2 数据验证

可参考:
https://www.cnblogs.com/jiangxifanzhouyudu/p/11107734.html
(1)基类抽象特性

using System;namespace MyAttribute.ValidateExtend
{public abstract class AbstractValidateAttribute : Attribute{public abstract bool Validate(object oValue);}
}

(2)子类特性实现–数字长度

using System;namespace MyAttribute.ValidateExtend
{[AttributeUsage(AttributeTargets.Property)]public class LongAttribute : AbstractValidateAttribute{private long _Min = 0;private long _Max = 0;public LongAttribute(long min, long max){this._Min = min;this._Max = max;}public override bool Validate(object oValue){return oValue != null&& long.TryParse(oValue.ToString(), out long lValue)&& lValue >= this._Min&& lValue <= this._Max;}}
}

(3)子类特性实现–可空

namespace MyAttribute.ValidateExtend
{public class RequiredAttribute : AbstractValidateAttribute{public override bool Validate(object oValue){return oValue != null&& !string.IsNullOrWhiteSpace(oValue.ToString());}}
}

(4)子类特性实现–字符串长度

using System;namespace MyAttribute.ValidateExtend
{[AttributeUsage(AttributeTargets.Property)]public class StringLengthAttribute : AbstractValidateAttribute{private int _Min = 0;private int _Max = 0;public StringLengthAttribute(int min, int max){this._Min = min;this._Max = max;}public override bool Validate(object oValue){return oValue != null&& oValue.ToString().Length >= this._Min&& oValue.ToString().Length <= this._Max;}}
}

(5)泛型扩展方法

using System;namespace MyAttribute.ValidateExtend
{public static class AttributeExtend{public static bool Validate<T>(this T t){Type type = t.GetType();foreach (var prop in type.GetProperties()){if (prop.IsDefined(typeof(AbstractValidateAttribute), true)){object oValue = prop.GetValue(t);foreach (AbstractValidateAttribute attribute in prop.GetCustomAttributes(typeof(AbstractValidateAttribute), true)){if (!attribute.Validate(oValue))return false;}}}return true;}}
}

(6)常规类字段定义

using System;namespace MyAttribute.ValidateExtend
{public static class AttributeExtend{public static bool Validate<T>(this T t){Type type = t.GetType();foreach (var prop in type.GetProperties()){if (prop.IsDefined(typeof(AbstractValidateAttribute), true)){object oValue = prop.GetValue(t);foreach (AbstractValidateAttribute attribute in prop.GetCustomAttributes(typeof(AbstractValidateAttribute), true)){if (!attribute.Validate(oValue))return false;}}}return tue;}}
}

(7)类调用扩展方法验证字段

using MyAttribute.EnumExtend;
using MyAttribute.ValidateExtend;
using System;namespace MyAttribute
{/// <summary>/// main方法调用/// </summary>class Program{static void Main(string[] args){try{                #region 特性实现数据验证,并且可扩展{//通过特性去提供额外行为//数据验证--到处都需要验证StudentVip student = new StudentVip(){Id = 123,Name = "无为",QQ = 729220650,Salary = 1010000};                    if (student.Validate()){Console.WriteLine("特性校验成功");}//1 可以校验多个属性//2 支持多重校验//3 支持规则的随意扩展}#endregion}catch (Exception ex){Console.WriteLine(ex.Message);}Console.Read();}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/15010.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

聊聊Python中的文件读写操作

Python 中的文件读写操作是数据处理和存储的基本操作之一。下面&#xff0c;我将详细解释如何在 Python 中进行文件的读写操作。 1. 打开文件 在 Python 中&#xff0c;使用 open() 函数来打开文件。这个函数需要至少一个参数&#xff0c;即文件名&#xff0c;并且返回一个文件…

数据结构和组成

数据结构组成 数据项&#xff1a;一个数据元素可以由若干数据项组成。 数据对象&#xff1a;有相同性质的数据元素的集合&#xff0c;是数据的子集。 数据结构&#xff1a;是相互之间存在一种或多种特定关系的数据元素的集合。 逻辑结构 数据对象中数据元素之间的相互关系 eg: …

深入理解python列表遍历:两种方法详解与实例

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、引言 二、使用索引遍历列表 三、直接使用元素遍历列表 四、总结 一、引言 在编程过程…

创建python字典的两种方法:直观与函数式

新书上架~&#x1f447;全国包邮奥~ python实用小工具开发教程http://pythontoolsteach.com/3 欢迎关注我&#x1f446;&#xff0c;收藏下次不迷路┗|&#xff40;O′|┛ 嗷~~ 目录 一、直观创建法&#xff1a;直接定义键值对 二、函数式创建法&#xff1a;使用内置函数dict…

CSRF 攻击

概述 CSRF(Cross-site request forgery,跨站请求伪造)。 它是指攻击者利用了用户的身份信息&#xff0c;执行了用户非本意的操作。 它首先引导用户访问一个危险网站&#xff0c;当用户访问网站后&#xff0c;网站会发送请求到被攻击的站点&#xff0c;这次请求会携带用户的c…

拼多多携手中国农业大学,投建陕西佛坪山茱萸科技小院

5月16日下午&#xff0c;中国农业大学陕西佛坪山茱萸科技小院在佛坪县银厂沟村揭牌。佛坪县素有“中国山茱萸之乡”的美誉&#xff0c;是全国山茱萸三大基地之一&#xff0c;当地山茱萸是国家地理标志产品&#xff0c;山茱萸肉产量位居全国第二。 为充分发挥佛坪县得天独厚的山…

vscode快捷键mac快捷键

vscode快捷键 在 Visual Studio Code 中&#xff0c;可以使用「Ctrl D」快捷键来选中相同内容。还可以使用以下快捷键来操作&#xff1a; 向上/下选择相同内容 Alt ↑/↓ 选择所有相同内容 Ctrl Shift L mac中则是commandshiftL VSCode删除整行快捷键 CtrlShiftK mac中则…

局部放电试验变频电源

局部放电试验中的变频电源设备 局部放电试验变频电源是一种专为电力设备的局部放电检测设计的高性能电源系统。在电力设备的运行和维护过程中&#xff0c;局部放电测试用于探测潜在的绝缘缺陷&#xff0c;防止它们进一步恶化导致设备损坏。传统的局部放电试验通常使用交流电源&…

with关键字

在 Python 中&#xff0c;with 是一个关键字&#xff0c;用于引入一个上下文管理器&#xff08;context manager&#xff09;。上下文管理器是一种特殊的对象&#xff0c;它允许你以一种干净、结构化的方式执行一组特定的操作&#xff0c;通常包括设置和清理资源。 with 语句通…

jmeter之测试计划

一、测试计划作用 测试计划是jmeter的默认控件所有线程组都是测试计划的下级控件测试计划可以配置用户自定义的变量测试计划可以配置线程组的串行或并行 二、查看界面 名称&#xff1a;可以修改自定义的名称注释&#xff1a;解释测试计划是用来做什么的用户自定义的变量&…

Sentinel重要的前置知识

文章目录 1、雪崩问题及解决方案1.1、雪崩问题1.2、超时处理1.3、仓壁模式1.4、断路器1.5、限流1.6、总结 2、服务保护技术对比3、Sentinel介绍和安装3.1、初识Sentinel3.2、安装Sentinel 4、微服务整合Sentinel ​&#x1f343;作者介绍&#xff1a;双非本科大三网络工程专业在…

PX4使用yolo仿真环境搭建

文章目录 前言一、修改机架sdf文件二、安装yolo三、运行 前言 ubuntu20.04 PX4 1.13.3 已配置好PX4 ROS gazebo环境 一、修改机架sdf文件 将双目相机加到仿真的iris机架上 修改下图文件 添加如下&#xff1a; <include><uri>model://stereo_camera</uri>…

用nn.Sequential实现图像的数据增强(augmentations)

代码example: import torch import torch.nn as nn# 定义一些增强操作&#xff0c;例如随机水平翻转和归一化 augmentations nn.Sequential(nn.RandomHorizontalFlip(),nn.Normalize(mean[0.5], std[0.5]) )# 创建一个示例 tensor candidate torch.randn(1, 3, 224, 224) #…

QWRT改AP模式 自动获取IP

关闭&#xff08;禁用&#xff09;WAN 和 WAN6接口修改LAN接口 IP地址 改为上级路由的网段&#xff0c;如主路由器IP192.168.2.1&#xff0c;那么就设置为192.168.2.2&#xff08;不要冲突了&#xff09;IPv4 网关 改为上级路由的IP地址&#xff0c;如主路由器IP192.168.2.1&am…

使用 CapSolver API 服务解决 Arkose Labs FunCaptcha 验证码

使用 CapSolver API 服务解决 Arkose Labs FunCaptcha 验证码 FunCaptcha 以其复杂的图像验证而闻名&#xff0c;对自动化系统构成了巨大的挑战。CapSolver 的 API 服务利用先进的 AI 技术轻松应对和解决 FunCaptcha 挑战。本指南探讨了 CapSolver 如何实现无缝自动化&#xff…

什么是html

HTML&#xff08;HyperText Markup Language&#xff0c;超文本标记语言&#xff09;是一种用于创建网页的标准标记语言。它描述了一个网站的结构骨架&#xff0c;使得浏览器能够展示具有特定格式的文本、链接、图片和其他内容。HTML 文档由一系列的元素构成&#xff0c;这些元…

STM32笔记-AD模数转换

目录 一、ADC介绍 二、ADC主要特征 三、ADC框图 1. ​​​​ 外部触发转换 ​ 2. 转换模式 3. 输入通道 4. 逻辑框图 四、校准 五、数据对齐 六、AD转换步骤 七、AD_Init(单通道AD转换)初始化函数配置 DMA: adc_dma_mode_enable(ADC0); 这段代码是用来使能ADC的DMA&a…

ts 字符串不能做索引异常提示 type because expression of type ‘string‘

Element implicitly has an any type because expression of type string cant be used to index type 例子 let a{b:"1",c:"1" } var b"b"; let ca[b] let ca[b]就会爆这个错误&#xff0c;因为在编译器看来b是一个未知的东西&#xff0c;它不…

什么是创造力?如何判断自己的创造力?

创造力&#xff0c;主要表现为创新思想、发现和创造新事物的能力&#xff0c;是知识&#xff0c;智力和能力的综合能力&#xff0c;尤其是在职业发展方面&#xff0c;创造力具有重要的意义&#xff0c;企业的核心竞争力就来源于创造力&#xff0c;这就需要具有创造力的员工来推…

ArduPilot开源飞控之MAVProxy深入研读系列 - 1基本操作

ArduPilot开源飞控之MAVProxy深入研读系列 - 1基本操作 1. 源由2. 基本操作2.1 二进制安装2.2 源代码安装2.3 硬链接飞控2.4 软连接飞控 3. 启动参数3.1 输入3.2 输出3.3 日志3.4 交互3.5 其他 4. 参考资料 1. 源由 玩开源&#xff0c;就尽量不要用Windows/Android/iOS/MaxOS什…