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

相关文章

2. c#从不同cs的文件调用函数

1.文件目录如下: 2. Program.cs文件的主函数如下 using System;using System.Collections.Generic;using System.Linq;using System.Threading.Tasks;using System.Windows.Forms;namespace datasAnalysis{internal static

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定

Java 创建图形用户界面(GUI)入门指南(Swing库 JFrame 类)概述

概述 基本概念 Java Swing 的架构 Java Swing 是一个为 Java 设计的 GUI 工具包,是 JAVA 基础类的一部分,基于 Java AWT 构建,提供了一系列轻量级、可定制的图形用户界面(GUI)组件。 与 AWT 相比,Swing 提供了许多比 AWT 更好的屏幕显示元素,更加灵活和可定制,具有更好的跨平台性能。 组件和容器 Java Swing 提供了许多

用命令行的方式启动.netcore webapi

用命令行的方式启动.netcore web项目 进入指定的项目文件夹,比如我发布后的代码放在下面文件夹中 在此地址栏中输入“cmd”,打开命令提示符,进入到发布代码目录 命令行启动.netcore项目的命令为:  dotnet 项目启动文件.dll --urls="http://*:对外端口" --ip="本机ip" --port=项目内部端口 例: dotnet Imagine.M

自定义类型:结构体(续)

目录 一. 结构体的内存对齐 1.1 为什么存在内存对齐? 1.2 修改默认对齐数 二. 结构体传参 三. 结构体实现位段 一. 结构体的内存对齐 在前面的文章里我们已经讲过一部分的内存对齐的知识,并举出了两个例子,我们再举出两个例子继续说明: struct S3{double a;int b;char c;};int mian(){printf("%zd\n",s

顺序表之创建,判满,插入,输出

文章目录 🍊自我介绍🍊创建一个空的顺序表,为结构体在堆区分配空间🍊插入数据🍊输出数据🍊判断顺序表是否满了,满了返回值1,否则返回0🍊main函数 你的点赞评论就是对博主最大的鼓励 当然喜欢的小伙伴可以:点赞+关注+评论+收藏(一键四连)哦~ 🍊自我介绍   Hello,大家好,我是小珑也要变强(也是小珑),我是易编程·终身成长社群的一名“创始团队·嘉宾”