C#泛型和反射实例解析
作者:shichen2014 发布时间:2023-10-04 03:49:02
标签:C#,泛型,反射
C#中的泛型和反射经常是一起工作的,因此这里就一次性的加以介绍了。
由于c#是强类型语言,一般来说函数的返回类型和参数的类型都是一早写好的,这也就造成了很多时候不像js那样方便使用,不够灵话。
因此就有了这个泛型,它可以让你的函数和参数在调用的时候才决定类型。如下例所示:
public T abc<T>(T word)
{
return word;
return default(T); //关键字default可以对引用类型返回nullAble,int类型返回0,初始化一个T的感觉啦
}
abc<string>("x");
//struct 是 值类型
//好处调用的是如果参数是值类型可以不用著名 test(100) 而不需要 test<int>(100);
public void test<T>(T number) where T : struct
{
int z = Convert.ToInt32(number);
//调用 test(100);
}
//下面的不知道好处在哪用在什么地方,
public void test2<T>(T lei) where T : class
{
}
public void test3<T>() where T : stooges
{
}
public T test4<T>() where T : new()
{
T abc = new T();
return abc;
}
public class stooges
{
}
加了where 我就不清楚在什么地方用的上了,这个以后再研究
反射能让我们的代码运行时动态的获取一些对象或者类的属性值等等,甚至是调用它们。
先来一个常用到的,我们想获取一个对象的全部属性和值, 用js 是:
for(var attr in object) { object[attr]=value, attr = attr }
var obj = new abc();
Type T = typeof(abc); //typeof(Class) 而不是 typeof(object) 哦
Type V = obj.GetType(); //obj.GetType() 就是typeof(object的class)
PropertyInfo[] attrs = obj.GetType().GetProperties(BindingFlags.Instance | BindingFlags.Public); //获取attrs
foreach (PropertyInfo attr in attrs)
{
string key = attr.Name; //获取attr name
object value = attr.GetValue(obj, null); //获取value
Type type = attr.PropertyType; //类型
}
关键就是那个 Type , 获取Type后就可以做很多了
C#常用的方法为:
T.GetProperty("key").GetValue(obj, null); //read a key value
T.GetProperty("key").SetValue(obj, "", null); //write a value to key
//注意如果是字典
T.GetProperty("Item").GetValue(obj, new [] {"id"}); //先拿Item 然后才通过 new[] {这里放指定的key}
再来看看更详细的
class MyClass
{
public int x { get; set; }
public int y { get; set; }
public MyClass(int i)
{
x = y + i;
}
public MyClass(int i, int j)
{
x = i;
y = j;
}
public int sum()
{
return x + y;
}
}
我们想获取这个Class 的构造函数 :
Type t = typeof(MyClass);
ConstructorInfo[] constructors = t.GetConstructors(); //使用这个方法获取构造函数列表
for (int i = 0; i < constructors.Length; i++)
{
ConstructorInfo constructor = constructors[i];
//构造函数也是方法所以有 GetParameters
ParameterInfo[] parameters = constructor.GetParameters(); //获取当前构造函数的参数列表
string paraTypeName = parameters[0].ParameterType.Name; //方法的参数类型名称
string paraName = parameters[0].Name;// 方法的参数名
}
//调用构造函数
object[] args = new object[2];
args[0] = 10;
args[1] = 20;
//不用new 直接实例化
object instance = constructors[0].Invoke(args); //实例化一个这个构造函数有两个参数的类型对象,如果参数为空,则为null
object instance = (t)Activator.CreateInstance(t); 还有这种实例的方法,不清楚可以放参数没有
调用方法如下:
MethodInfo[] methods = T.GetMethods(BindingFlags.Public | BindingFlags.DeclaredOnly | BindingFlags.Instance);
foreach (MethodInfo method in methods)
{
string return_name = method.ReturnType.Name; //返回方法的返回类型
string name = method.Name;
if (name.Equals("sum", StringComparison.Ordinal)) //指定方法名调用
{
int value = (int)method.Invoke(instance, null); //instance是之前实例好的对象,方法就是在这个对象之中
}
}
下面是一些参考代码:
Type t = typeof(MyClass);
Console.WriteLine("----------------Method------------------");
MethodInfo[] methods = t.GetMethods();
foreach (MethodInfo method in methods)
{
Console.WriteLine("Method:" + method);
//Console.WriteLine(method);
//Console.WriteLine("返回值:" + method.ReturnParameter);
}
Console.WriteLine("---------------Field-------------------"); //字段 ,比如这种 private static string name;
FieldInfo[] fields = t.GetFields(BindingFlags.NonPublic | BindingFlags.Instance | BindingFlags.Static);
foreach (FieldInfo field in fields)
{
Console.WriteLine("Field:" + field);
}
Console.WriteLine("--------------Member--------------------"); //成员即方法和属性
MemberInfo[] members = t.GetMembers();
foreach (MemberInfo member in members)
{
Console.WriteLine("Member:" + member);
}
Console.WriteLine("--------------Property--------------------"); //属性
PropertyInfo[] properties = t.GetProperties();
foreach (PropertyInfo property in properties)
{
Console.WriteLine("Property:" + property);
}
Console.WriteLine("--------------Constructor--------------------"); //构造函数
ConstructorInfo[] constructors = t.GetConstructors(BindingFlags.NonPublic | BindingFlags.Instance);
foreach (ConstructorInfo constructor in constructors)
{
Console.WriteLine("Constructor:" + constructor);
}
调用不用记太清楚,关键知道什么东西可以用反射获取和调用就可以了。
比较常使用的地方是用泛型写方法的时候,比如我们的泛型是一个不确定的类,然后我们要获取attr值等等的,就很好可以使用反射了。
还有通过string来调用方法,这个在js很长用到,比如 obj["methodName"](); 这里也可以用到反射来实现。
此外还需要注意: 反射的性能是很慢的,也可以说动态就是慢,这个是很正常的,反射的性能优化可以参考使用动态编程之类的,不过这里就不提了。


猜你喜欢
- 在Flutter * 一个某支付价值几个亿的页面这一篇中,我们使用了 ListView 将几个 GridVie
- 一、引言Good Good Study,Day Day Up,童鞋点个关注,不迷路,么么哒~~~MP自带的条件构造器虽然很强大,有时候也避免
- 按照常规思维,往一个map里put一个已经存在的key,会把原有的key对应的value值覆盖,然而通过一次线上问题,发现Java8中的Co
- 先给大家说下我实现的思路:在popouWindow里面加上ListView,数据是把List以字符串按照JSON的样式存入本地,先看看效果a
- 由于最近想要阅读下JDK1.8 中HashMap的具体实现,但是由于HashMap的实现中用到了红黑树,所以我觉得有必要先复习下红黑树的相关
- 简介Dijkstra(迪杰斯特拉)算法是典型的单源最短路径算法,用于计算一个节点到其他所有节点的最短路径。主要特点是以起始点为中心向外层层扩
- 微服务之间token传递问题假设现在有A服务,B服务,外部使用RESTApi请求调用A服务,在请求头上有token字段,A服务使用完后,B服
- 一、AOP概述AOP,即面向切面编程,简单来说就是将代码中重复的部分抽取出来,在需要执行的时候使用 * 的技术,在不修改源码的基础上对方法
- 因为课程需要,昨天好多同学在安装Android studio3.6.1后,无法构建,不知道什么原因,我的电脑上使用的是之前3.4版本的,可以
- 1. this 引用1.1 为什么要有this引用先来写一个日期类的例子:public class classCode { &
- 一、MyBatis的逆向⼯程(1)所谓的逆向⼯程是:根据数据库表逆向⽣成Java的pojo类,SqlMapper.xml⽂件,以及Mappe
- Java 8中引入了CompletableFuture类,它是一种方便的异步编程工具,可以处理各种异步操作,如网络请求、文件IO和数据库操作
- 本文实例讲述了Java递归算法。分享给大家供大家参考,具体如下:1.实现1到100的和,用递归实现public class Recursio
- settings.xml有什么用?如果在Eclipse中使用过Maven插件,想必会有这个经验:配置settings.xml文件的路径。se
- 1. 启动 Redis Server启动 redis server,如下图所示,端口号 6379:2. 工程实例2.1 工程目录工程目录如下
- 1、private实现封装处理如果像想要知道封装,首先必须清楚如果没有封装会怎么样?没有封装方法中的属性,在所有方法被调用后都可以进行无权限
- 一、导入外部DLL函数如[DllImport(“kernel32.dll”)]这叫引入kernel3
- 抽象类和抽象方法常用知识点:(1)抽象类作为被继承类,子类必须实现抽象类中的所有抽象方法,除非子类也为抽象类。也就是说,如果子类也为抽象类,
- Error:All flavors must now belong to a named flavor dimension. The fla
- instanceof关键字用于判断一个引用类型变量所指向的对象是否是一个类(或接口、抽象类、父类)的实例。 举个例子:public