反射实例化类

public class Person
{
public string Name { get; set; } public Person(string name)
{
this.Name = name;
} public string Say(string msg)
{
return $"{Name}: {msg}";
}
} class Program
{
// 测试次数
const int count = 10000000; static void Main(string[] args)
{
CreateInstance0();
CreateInstance1();
CreateInstance2();
CreateInstance3();
CreateInstance4(); Console.Read();
} static void CreateInstance0()
{
Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
Person person = new Person("张三");
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - new");
} static void CreateInstance1()
{
Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
object person = Activator.CreateInstance(typeof(Person), "张三");
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - Activator.CreateInstance");
} static void CreateInstance2()
{
Assembly assembly = Assembly.GetExecutingAssembly(); Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
Person obj = (Person)assembly.CreateInstance("ConsoleTest.Person", true, BindingFlags.Default, null, new object[] { "张三" }, null, null);
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - Assembly.CreateInstance");
} static void CreateInstance3()
{
Assembly assembly = Assembly.GetExecutingAssembly(); Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
Type type = assembly.GetType("ConsoleTest.Person");
object person = Activator.CreateInstance(type, "张三");
}
watch.Stop();
Console.WriteLine($"{watch.Elapsed} - Assembly.CreateInstance1");
} static void CreateInstance4()
{
Assembly assembly = Assembly.GetExecutingAssembly(); Stopwatch watch = new Stopwatch();
watch.Start(); Type type = assembly.GetType("ConsoleTest.Person");
for (var i = 0; i < count; i++)
{
object person = Activator.CreateInstance(type, "张三");
}
watch.Stop();
Console.WriteLine($"{watch.Elapsed} - Assembly.CreateInstance2");
}
}

  • 通过反射实例化对象,要比直接 new 要慢 50 倍左右
  • assembly.CreateInstance 要比 Activator.CreateInstance 慢,主要的性能损耗在 Assembly.GetType

反射调用类的方法

class Program
{
// 测试次数
const int count = 10000000; static void Main(string[] args)
{
InvokeMethod0();
InvokeMethod1();
InvokeMethod2();
InvokeMethod3();
InvokeMethod4(); Console.Read();
} static void InvokeMethod0()
{
Person person = new Person("张三"); Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
string name = person.Say("Hello World!");
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - 直接调用");
} static void InvokeMethod1()
{
Person person = (Person)Activator.CreateInstance(typeof(Person), "张三"); Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
string name = person.Say("Hello World!");
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - 反射缓存类调用");
} static void InvokeMethod2()
{
Person person = (Person)Activator.CreateInstance(typeof(Person), "张三");
MethodInfo method = typeof(Person).GetMethod(nameof(Person.Say), new Type[] { typeof(string) });
Func<string, string> func = (Func<string, string>)method.CreateDelegate(typeof(Func<string, string>), person); Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
string result = func("Hello World!");
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - 使用反射创建出来的委托调用");
} static void InvokeMethod3()
{
Person person = (Person)Activator.CreateInstance(typeof(Person), "张三"); MethodInfo method = typeof(Person).GetMethod(nameof(Person.Say), new Type[] { typeof(string) }); object[] parameters = new object[] { "Hello World!" }; Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
string name = (string)method.Invoke(person, parameters);
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - 使用反射得到的方法缓存调用");
} static void InvokeMethod4()
{
Person person = (Person)Activator.CreateInstance(typeof(Person), "张三"); object[] parameters = new object[] { "Hello World!" }; Stopwatch watch = new Stopwatch();
watch.Start(); for (var i = 0; i < count; i++)
{
string result = (string)(typeof(Person).GetMethod(nameof(Person.Say))?.Invoke(person, parameters));
} watch.Stop();
Console.WriteLine($"{watch.Elapsed} - 直接使用反射调用");
}
}

  • 反射得到实例后调用方法和直接调用方法效率一样
  • 缓存反射方法调用和直接使用反射调用都非常耗效率

【C#】反射的用法及效率对比的更多相关文章

  1. c#中@标志的作用 C#通过序列化实现深表复制 细说并发编程-TPL 大数据量下DataTable To List效率对比 【转载】C#工具类:实现文件操作File的工具类 异步多线程 Async .net 多线程 Thread ThreadPool Task .Net 反射学习

    c#中@标志的作用   参考微软官方文档-特殊字符@,地址 https://docs.microsoft.com/zh-cn/dotnet/csharp/language-reference/toke ...

  2. golang 浮点数 取精度的效率对比

    需求 浮点数取2位精度输出 实现 代码 package main import ( "time" "log" "strconv" " ...

  3. string中Insert与Format效率对比、String与List中Contains与IndexOf的效率对比

    关于string的效率,众所周知的恐怕是“+”和StringBuilder了,这些本文就不在赘述了.关于本文,请先回答以下问题(假设都是基于多次循环反复调用的情况下):1.使用Insert与Forma ...

  4. 关于 pgsql 数据库json几个函数用法的效率测试

    关于 pgsql 数据库json几个函数用法的效率测试 关于pgsql 几个操作符的效率测试比较1. json::->> 和 ->> 测试方法:单次运行100次,运行10个单次 ...

  5. 转 SQL Union和SQL Union All两者用法区别效率以及与order by 和 group by配合问题

    SQL Union和SQL Union All两者用法区别效率以及与order by 和 group by配合问题 SQL Union和SQL Union All用法 SQL UNION 操作符 UN ...

  6. FileInputStream 与 BufferedInputStream 效率对比

    我的技术博客经常被流氓网站恶意爬取转载.请移步原文:http://www.cnblogs.com/hamhog/p/3550158.html ,享受整齐的排版.有效的链接.正确的代码缩进.更好的阅读体 ...

  7. java中多种写文件方式的效率对比实验

    一.实验背景 最近在考虑一个问题:“如果快速地向文件中写入数据”,java提供了多种文件写入的方式,效率上各有异同,基本上可以分为如下三大类:字节流输出.字符流输出.内存文件映射输出.前两种又可以分为 ...

  8. SIMD---SSE系列及效率对比

    SSE(即Streaming SIMD Extension),是对由MMX指令集引进的SIMD模型的扩展.我们知道MMX有两个明显的缺点: 只能操作整数. 不能与浮点数同时运行(MMX使用FPU寄存器 ...

  9. Snapman系统中TCC执行效率和C#执行效率对比

    Snapman集合了TCC编译器可以直接编译执行C语言脚本,其脚本执行效率和C#编译程序进行效率对比,包括下面4方面: 1.函数执行效率 2.数字转换成字符串 3.字符串的叠加 4.MD5算法 这是C ...

随机推荐

  1. ASP.NET Core 5.0 MVC中的视图分类——布局视图、启动视图、具体视图、分部视图

    一.创建MVC应用程序 创建后的项目 二.(全局性)启动视图 _ViewStart.cshtml 顾名思义,就是在View开始执行之前执行,而且是每一个View, 它的预设内容是 @{ Layout ...

  2. 向指定url发送Get/Post请求

    向指定url发送Get/Post请求 1.向指定url发送Get/Post请求 2.HttpUtil 工具类–向指定url发送Get/Post请求 1.向指定url发送Get/Post请求 impor ...

  3. 最新Ceph L版与openstack Pike对接

    安装Ceph luminous   实验环境 三台服务器,每台服务器都有4块硬盘,每台服务器都将自己的第一块硬盘作为系统盘,剩下的做ceph   一.在所有服务器上操作 #使用阿里源 yum inst ...

  4. vmware打开虚拟级断电情况下,无法找到虚拟机文件

    1.此时会在建立的虚拟机目录下,有一些 %虚拟机名字%.vmx.lck 或者别的   %虚拟机名字%.***.lck   删除这些文件夹 2.虚拟文件 是一个后缀名为vmx的文件,发现断电后 变成了v ...

  5. Birkhoff-von Neumann Crossbar 光交换网络的调度方案

    Birkhoff-von Neumann Crossbar 光交换网络的调度方案 ​ This is a summary aimed at looking for "high perform ...

  6. 如何安装Python 3.9.1?

    首先打开浏览器输入网址:https://www.python.org或者通过百度搜索python进入Python官网. 选择Downloads,弹出最新版本下载链接,当前版本为3.9.1,如图所示: ...

  7. MyBatis逆向工程生成dao层增删改查方法解释使用(转载)

    int countByExample(BUserExample example); //根据条件查询数量 /** * 示例 * public int countByExample() { * BUse ...

  8. poj 3304 Segments(解题报告)

    收获:举一反三:刷一道会一道 1:思路转化:(看的kuangbin的思路) 首先是在二维平面中:如果有很多线段能够映射到这个直线上并且至少重合于一点,充要条件: 是过这个点的此条直线的垂线与其他所有直 ...

  9. hdu5531 Rebuild

    Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 262144/262144 K (Java/Others) Total Submissi ...

  10. Codeforces Round #494 (Div. 3) D. Coins and Queries (贪心,数学)

    题意:给你一组全是\(2^d\ (d\ge0)\)的数,询问q次,每次询问一个数,问这个数是否能够由原数组中的数相加得到,如果能,输出最少用多少个数,否则输出\(-1\). 题解:首先贪心得出结论:如 ...