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

相关文章

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. 使

Python中使用uv创建环境及原理举例详解

《Python中使用uv创建环境及原理举例详解》uv是Astral团队开发的高性能Python工具,整合包管理、虚拟环境、Python版本控制等功能,:本文主要介绍Python中使用uv创建环境及... 目录一、uv工具简介核心特点:二、安装uv1. 通过pip安装2. 通过脚本安装验证安装:配置镜像源(可

Springboot3+将ID转为JSON字符串的详细配置方案

《Springboot3+将ID转为JSON字符串的详细配置方案》:本文主要介绍纯后端实现Long/BigIntegerID转为JSON字符串的详细配置方案,s基于SpringBoot3+和Spr... 目录1. 添加依赖2. 全局 Jackson 配置3. 精准控制(可选)4. OpenAPI (Spri

MySQL JSON 查询中的对象与数组技巧及查询示例

《MySQLJSON查询中的对象与数组技巧及查询示例》MySQL中JSON对象和JSON数组查询的详细介绍及带有WHERE条件的查询示例,本文给大家介绍的非常详细,mysqljson查询示例相关知... 目录jsON 对象查询1. JSON_CONTAINS2. JSON_EXTRACT3. JSON_TA

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

C#如何去掉文件夹或文件名非法字符

《C#如何去掉文件夹或文件名非法字符》:本文主要介绍C#如何去掉文件夹或文件名非法字符的问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C#去掉文件夹或文件名非法字符net类库提供了非法字符的数组这里还有个小窍门总结C#去掉文件夹或文件名非法字符实现有输入字

C#之List集合去重复对象的实现方法

《C#之List集合去重复对象的实现方法》:本文主要介绍C#之List集合去重复对象的实现方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C# List集合去重复对象方法1、测试数据2、测试数据3、知识点补充总结C# List集合去重复对象方法1、测试数据

C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式

《C#实现将Office文档(Word/Excel/PDF/PPT)转为Markdown格式》Markdown凭借简洁的语法、优良的可读性,以及对版本控制系统的高度兼容性,逐渐成为最受欢迎的文档格式... 目录为什么要将文档转换为 Markdown 格式使用工具将 Word 文档转换为 Markdown(.