C# 根据typename字符串,创建任意类型的对象(含泛型)

2024-02-25 20:38

本文主要是介绍C# 根据typename字符串,创建任意类型的对象(含泛型),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

认识一下反射

使用反射得到任意类型

动态创建实例

通过类名获得类型

分析如何获得泛型类型

实现根据类名,获得泛型类型

小计


认识一下反射

在c#中,反射时个很好用的东西,可以通过反射动态创建一个实例,且实例类型不唯一

一个很简单的例子,来了解下反射可以做什么

Type t = typeof(Form);
dynamic frm = t.Assembly.CreateInstance(t.FullName);
// 以上内容完全等同于
dynamic frm = new Form();Type t = typeof(List<string>);
dynamic lst = t.Assembly.CreateInstance(t.FullName);
// 以上内容完全等同于
dynamic lst = new List<string>();

看起来好像反射没什么用,因为他现在完成的,我们都可以自己完成啊

那么,换个场景,有一个函数,传递进来一个object对象,如果对象存在Run方法,就运行该方法,如果不存在Run,但存在Start,则运行Start方法,再不行,就按照Begin、Do等方法依次向后查找,这个时候,如果用switch来枚举object可能的类型不是不可以,但是这样会造成后期需要经常维护此处代码,所以用反射来调用

public void RefInvokeStart(object obj){Type t = obj.GetType();string[] methods = new string[]{"Run","Start","Begin","Do"};for(int i=0;i<methods.Length;i++){MethodInfo method = t.GetMethod(methods[i]);if (method!=null){try{method.Invoke(null,new object[]{obj,null});return;}catch(Exception ex){}}}
}

以上是反射的简单例子,现在翻篇了。

----------------------------------

使用反射得到任意类型

在实际应用中,有时候我们在开发的过程中,脑子会突然抽筋,非常想不开的想用一个方法,来实现不同的需求,嗯,泛型就是为此准备的,但实际工作时,我们只知道对象类型,甚至只知道类型名称,连类型全名(含命名空间路径的类型名称)都不知道,就要创建相应类型的实例

嗯。。。。其实文盲脑子也这么抽过,虽然后来回头看的时候,总觉得不需要这么处理,还有其他方法。。。。但脑子抽的时候可没想过这些 T_T

动态创建实例

先来个根据已知对象动态创建实例的方法

public static T CreateElement<T>(){Type t = typeof(T);return (T)t.Assembly.CreateInstance(t.FullName);
}// 很简单的就可以创建,基本等同于newList<string> ls = CreateElement<List<string>>();

通过类名获得类型

然后,我们只知道类型的名字,不知道全名,也没有确定类型。。。

那么,我们就需要先得到一个Type才可以

        public static Type GetTypeByName(string typename){Type t = null;string source = typename;try{t = Type.GetType(source);if (t != null){return t;}Assembly[] assembly = AppDomain.CurrentDomain.GetAssemblies();foreach (Assembly ass in assembly){t = ass.GetType(source);if (t != null){return t;}Type[] ts = ass.GetTypes();foreach (Type st in ts){if (RegexExpand.IsMatch(st.FullName, @"\." + RegexExpand.FormatRegExp(source) + @"(`?\d+)?$")){return st;}}}}catch (Exception ex){}return t;}

简单来讲,就是根据typename字符串,获得Type类型数据,根据当前项目所调用的所有类库中,查找相同,或者包含我们写的这个名称的类型,并返回,比如 Type t1 = GetTypeByName("string");GetTypeByName("Form");GetTypeByName("Thread");

分析如何获得泛型类型

嗯,简单的名称对照类型就这么简单。再然后,就是重点了,有很多类型是支持泛型的,这些东西能通过反射得到类型吗?

比如,我现在想创建一个 Dictionary<string,Dictionary<int,List<Thread>>>对象,想用 GetTypeByName("Dictionary<string,Dictionary<int,List<Thread>>>")来尝试一下能否获得Type。嗯,不出预料,返回了一个null

因为我们所有对比的内容,都是单一类型,而传递进去的这个字符串,却是一个多类型组合的多泛型结构。。。还tmd带嵌套的

但是,也不是不能实现,整理下思路:

首先,最内层一对<>里肯定不包含其他类型了,这个是已经确定的;

其次,我们肯定不能写一个多泛型的对象,里面多个泛型都还是支持泛型的,比如绝对不会出现Dictionary<List<string>,List<int>>这样的两个泛型都是支持泛型的类型。。有一个就够了

再次,我们根据<>,从里到外获得所有类型字符串,并按顺序记录下来,然后将内层的对象作为参数传递给外层的对象,通过MakeGenericType方法来创建支持泛型的对象

实现根据类名,获得泛型类型

那么,思路有了,接下来就来实现这个方法吧

        public static Type GetTypeByName(string typename){Type t = null;string source = typename;if (source.IndexOf('<') > 0){List<string> lv = new List<string>();while (RegexExpand.IsMatch(source, @"<[^<>]+>")){lv.Add(RegexExpand.Match(source, @"(?<=<)[^<>]+(?=>)").Value);source = RegexExpand.Replace(source, @"<[^<>]+>", "/" + (lv.Count - 1));}List<Type[]> args = new List<Type[]>();for (int i = 0; i < lv.Count; i++){List<Type> arg = new List<Type>();string[] sp = lv[i].Split(',');for (int j = 0; j < sp.Length; j++){string s = sp[j].Trim();if (!string.IsNullOrEmpty(s)){if (RegexExpand.IsMatch(s, @"/\d+$")){Match m = RegexExpand.Match(s, @"^([^/\s]+)\s*/(\d+)$");if (!m.Success){throw new Exception("");}Type p = GetTypeByName(m.Groups[1].Value);Type c = p.MakeGenericType(args[Convert.ToInt32(m.Groups[2].Value)]);arg.Add(c);}else{arg.Add(GetTypeByName(s));}}}args.Add(arg.ToArray());}Match f = RegexExpand.Match(source, @"^([^/\s]+)\s*/(\d+)$");if (!f.Success){throw new Exception("");}Type fp = GetTypeByName(f.Groups[1].Value);Type fc = fp.MakeGenericType(args[Convert.ToInt32(f.Groups[2].Value)]);return fc;}else{try{t = Type.GetType(source);if (t != null){return t;}Assembly[] assembly = AppDomain.CurrentDomain.GetAssemblies();foreach (Assembly ass in assembly){t = ass.GetType(source);if (t != null){return t;}Type[] ts = ass.GetTypes();foreach (Type st in ts){if (RegexExpand.IsMatch(st.FullName, @"\." + RegexExpand.FormatRegExp(source) + @"(`?\d+)?$")){return st;}}}}catch (Exception ex){}}return t;}

得到类型了,创建实例就跟简单了,将之前出现的CreateElement变动一下

public static dynamic CreateElement(string typename){Type t = GetTypeByName(typename);return t.Assembly.CreateInstance(t.FullName);
}

最后就是创建实例对象了

dynamic obj = CreateElement("Dictionary<string,Dictionary<int,List<Thread>>>");

bingo,成了!

-------------

小计

结束语:虽然这个东西确实做出来了,但以我多次脑抽的经验来看,用的上这个方法的人,多半处于脑抽状态

这篇关于C# 根据typename字符串,创建任意类型的对象(含泛型)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/746632

相关文章

C# 比较两个list 之间元素差异的常用方法

《C#比较两个list之间元素差异的常用方法》:本文主要介绍C#比较两个list之间元素差异,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录1. 使用Except方法2. 使用Except的逆操作3. 使用LINQ的Join,GroupJoin

Python实现对阿里云OSS对象存储的操作详解

《Python实现对阿里云OSS对象存储的操作详解》这篇文章主要为大家详细介绍了Python实现对阿里云OSS对象存储的操作相关知识,包括连接,上传,下载,列举等功能,感兴趣的小伙伴可以了解下... 目录一、直接使用代码二、详细使用1. 环境准备2. 初始化配置3. bucket配置创建4. 文件上传到os

MySQL查询JSON数组字段包含特定字符串的方法

《MySQL查询JSON数组字段包含特定字符串的方法》在MySQL数据库中,当某个字段存储的是JSON数组,需要查询数组中包含特定字符串的记录时传统的LIKE语句无法直接使用,下面小编就为大家介绍两种... 目录问题背景解决方案对比1. 精确匹配方案(推荐)2. 模糊匹配方案参数化查询示例使用场景建议性能优

MySQL 获取字符串长度及注意事项

《MySQL获取字符串长度及注意事项》本文通过实例代码给大家介绍MySQL获取字符串长度及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 获取字符串长度详解 核心长度函数对比⚠️ 六大关键注意事项1. 字符编码决定字节长度2

python如何创建等差数列

《python如何创建等差数列》:本文主要介绍python如何创建等差数列的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python创建等差数列例题运行代码回车输出结果总结python创建等差数列import numpy as np x=int(in

怎么用idea创建一个SpringBoot项目

《怎么用idea创建一个SpringBoot项目》本文介绍了在IDEA中创建SpringBoot项目的步骤,包括环境准备(JDK1.8+、Maven3.2.5+)、使用SpringInitializr... 目录如何在idea中创建一个SpringBoot项目环境准备1.1打开IDEA,点击New新建一个项

如何使用Maven创建web目录结构

《如何使用Maven创建web目录结构》:本文主要介绍如何使用Maven创建web目录结构的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录创建web工程第一步第二步第三步第四步第五步第六步第七步总结创建web工程第一步js通过Maven骨架创pytho

MySQL 用户创建与授权最佳实践

《MySQL用户创建与授权最佳实践》在MySQL中,用户管理和权限控制是数据库安全的重要组成部分,下面详细介绍如何在MySQL中创建用户并授予适当的权限,感兴趣的朋友跟随小编一起看看吧... 目录mysql 用户创建与授权详解一、MySQL用户管理基础1. 用户账户组成2. 查看现有用户二、创建用户1. 基

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使