C# 使用绑定句柄来减少进程的内存耗用

许多应用程序中,绑定了一组类型(type)或者类型成员(从memberinfo派生),并将这些对象保存在某种形式的一个集合中。以后,会搜索这个集合,查找特定的对象,然后调用这个对象。这是一个很好的机制,但是有个小问题:type和memberinfo派生的对象需要大量的内存。如果一个应用程序容纳了太多这样的类,但只是偶尔用一下它们,应用程序的内存就会急剧增长,对应用程序的性能产生影响。

在内部,CLR用一种更精简的形式来表示这种信息。CLR之所以为应用程序创建这些对象,只是为了简化开发人员的工作。CLR在运行时并不需要这些大对象。如果需要缓存大量Type和MemberInfo派生对象,开发人员可以使用运行时句柄(runtime handle)来代替对象,从而减少工作集(占用的内存)。FCL定义了3个运行时句柄类型(都在System命名空间中),RuntimeTypeHandle,RuntimeFieldHandle,RumtimeMethodHandle。三个类型都是值类型,他们只包含了一个字段,也就是一个IntPtr;这样一来,这些类型的实例就相当省内存。ItPtr字段是一个句柄,它引用了AppDomain的Loader堆中的一个类型,字段或方法。转换方法:

Type→RuntimeTypeHandle,通过查询Type的只读字段属性TypeHandle。

RuntimeTypeHandle→Type,通过调用Type的静态方法GetTypeFromHanlde。

FieldInfo→RuntimeFieldHandle,通过查询FieldInfo的实例只读字段FieldHandle。

RuntimeFieldHandle→FieldInfo,通过调用FieldInfo的静态方法GetFieldFromHandle。

MethodInfo→RuntimeMethodHandle,通过查询MethodInof的实例只读字段MethodHandle。

RuntimeMethodHandle→MethodInfo,通过调用MethodInfo的静态方法GetMethodFromHandle。

下面的示例获取许多的MethodInfo对象,把它们转化成RuntimeMethodHandle实例,并演示转换前后的内存差异。

 private void UseRuntimeHandleToReduceMemory()        {            Show("Before doing anything");//从MSCorlib.dll中地所有方法构建methodInfos 对象缓存            List methodInfos = new List();                        foreach (Type t in typeof(object).Assembly.GetExportedTypes())            {                if (t.IsGenericType) continue;                MethodBase[] mbs = t.GetMethods(c_bf);                methodInfos.AddRange(mbs);            }            //显示当绑定所有方法之后,方法的个数和堆的大小            Console.WriteLine("# of Methods={0:###,###}", methodInfos.Count);            Show("After building cache of MethodInfo objects");//为所有MethodInfo对象构建RuntimeMethodHandle缓存            List methodHandles = new List();            methodHandles = methodInfos.ConvertAll(m => m.MethodHandle);            Show("Holding MethodInfo and RuntimeMethodHandle");            GC.KeepAlive(methodHandles);//阻止缓存被过早垃圾回收            methodInfos = null;//现在允许缓存垃圾回收            Show("After freeing MethodInfo objects");            methodInfos = methodHandles.ConvertAll(r => MethodBase.GetMethodFromHandle(r));            Show("Size of heap after re-creating methodinfo objects");            GC.KeepAlive(methodHandles);//阻止缓存被过早垃圾回收            GC.KeepAlive(methodInfos);//阻止缓存被过早垃圾回收            methodInfos = null;//现在允许缓存垃圾回收            methodHandles = null;//现在允许缓存垃圾回收            Show("after freeing MethodInfo and MethodHandle objects");        }

结果如下:

Heap Size =     114,788 - Before doing anything# of Methods=10,003Heap Size =   2,205,652 - After building cache of MethodInfo objectsHeap Size =   2,245,744 - Holding MethodInfo and RuntimeMethodHandleHeap Size =   2,171,976 - After freeing MethodInfo objectsHeap Size =   2,327,516 - Size of heap after re-creating methodinfo objectsHeap Size =     247,028 - after freeing MethodInfo and MethodHandle objects

本文整理自《NET CLR via C#》

作者:jiankunking 出处:http://www.php.cn/

许多应用程序中,绑定了一组类型(type)或者类型成员(从memberinfo派生),并将这些对象保存在某种形式的一个集合中。以后,会搜索这个集合,查找特定的对象,然后调用这个对象。这是一个很好的机制,但是有个小问题:type和memberinfo派生的对象需要大量的内存。如果一个应用程序容纳了太多这样的类,但只是偶尔用一下它们,应用程序的内存就会急剧增长,对应用程序的性能产生影响。

在内部,CLR用一种更精简的形式来表示这种信息。CLR之所以为应用程序创建这些对象,只是为了简化开发人员的工作。CLR在运行时并不需要这些大对象。如果需要缓存大量Type和MemberInfo派生对象,开发人员可以使用运行时句柄(runtime handle)来代替对象,从而减少工作集(占用的内存)。FCL定义了3个运行时句柄类型(都在System命名空间中),RuntimeTypeHandle,RuntimeFieldHandle,RumtimeMethodHandle。三个类型都是值类型,他们只包含了一个字段,也就是一个IntPtr;这样一来,这些类型的实例就相当省内存。ItPtr字段是一个句柄,它引用了AppDomain的Loader堆中的一个类型,字段或方法。转换方法:

Type→RuntimeTypeHandle,通过查询Type的只读字段属性TypeHandle。

RuntimeTypeHandle→Type,通过调用Type的静态方法GetTypeFromHanlde。

FieldInfo→RuntimeFieldHandle,通过查询FieldInfo的实例只读字段FieldHandle。

RuntimeFieldHandle→FieldInfo,通过调用FieldInfo的静态方法GetFieldFromHandle。

MethodInfo→RuntimeMethodHandle,通过查询MethodInof的实例只读字段MethodHandle。

RuntimeMethodHandle→MethodInfo,通过调用MethodInfo的静态方法GetMethodFromHandle。

下面的示例获取许多的MethodInfo对象,把它们转化成RuntimeMethodHandle实例,并演示转换前后的内存差异。

 private void UseRuntimeHandleToReduceMemory()        {            Show("Before doing anything");//从MSCorlib.dll中地所有方法构建methodInfos 对象缓存            List methodInfos = new List();                        foreach (Type t in typeof(object).Assembly.GetExportedTypes())            {                if (t.IsGenericType) continue;                MethodBase[] mbs = t.GetMethods(c_bf);                methodInfos.AddRange(mbs);            }            //显示当绑定所有方法之后,方法的个数和堆的大小            Console.WriteLine("# of Methods={0:###,###}", methodInfos.Count);            Show("After building cache of MethodInfo objects");//为所有MethodInfo对象构建RuntimeMethodHandle缓存            List methodHandles = new List();            methodHandles = methodInfos.ConvertAll(m => m.MethodHandle);            Show("Holding MethodInfo and RuntimeMethodHandle");            GC.KeepAlive(methodHandles);//阻止缓存被过早垃圾回收            methodInfos = null;//现在允许缓存垃圾回收            Show("After freeing MethodInfo objects");            methodInfos = methodHandles.ConvertAll(r => MethodBase.GetMethodFromHandle(r));            Show("Size of heap after re-creating methodinfo objects");            GC.KeepAlive(methodHandles);//阻止缓存被过早垃圾回收            GC.KeepAlive(methodInfos);//阻止缓存被过早垃圾回收            methodInfos = null;//现在允许缓存垃圾回收            methodHandles = null;//现在允许缓存垃圾回收            Show("after freeing MethodInfo and MethodHandle objects");        }

结果如下:

Heap Size =     114,788 - Before doing anything# of Methods=10,003Heap Size =   2,205,652 - After building cache of MethodInfo objectsHeap Size =   2,245,744 - Holding MethodInfo and RuntimeMethodHandleHeap Size =   2,171,976 - After freeing MethodInfo objectsHeap Size =   2,327,516 - Size of heap after re-creating methodinfo objectsHeap Size =     247,028 - after freeing MethodInfo and MethodHandle objects

以上就是C# 使用绑定句柄来减少进程的内存耗用的内容,更多相关内容请关注PHP中文网(www.php.cn)!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432238.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 06:23:45
下一篇 2025年12月17日 06:23:55

相关推荐

  • C# 反射详解

    以上就是C# 反射详解的内容,更多相关内容请关注PHP中文网(www.php.cn)!

    2025年12月17日
    000
  • C# 清除事件绑定的函数

    c#  清除事件绑定的函数 #region 清除事件绑定的函数 /// /// 清除事件绑定的函数 /// /// 拥有事件的实例 /// 事件名称 public static void ClearAllEvents(object objectHasEvents, string eventName)…

    好文分享 2025年12月17日
    000
  • C#中Encoding.Unicode与Encoding.UTF8的区别

    原文地址:点击打开链接 今天在园子首页看到一篇博文-简单聊下Unicode和UTF-8,从中知道了UTF-8是Unicode的一种实现方式: Unicode只是给这世界上每个字符规定了一个统一的二进制编号,并没有规定程序该如何去存储和解析。 可以说UTF-8是Unicode实现方式之一……

    2025年12月17日
    000
  • C# Protobuf-Net 序列化

    源码位置:protobuf-net 1、安装Nuget : 工具–拓展管理器 安装完成后重启Microsoft Visual Studio 2010,可以看到下图: 小注: 只有该解决方案已经打开项目的时候,才会看到下面两项: 2、安装protobuf_net(在Nuget中找到prot…

    2025年12月17日 好文分享
    000
  • C# 文件流压缩解压

    /// /// 文件流压缩解压 /// public class ZipHelper { public static int BEST_COMPRESSION = 9; public static int BEST_SPEED = 1; public static int DEFAULT_COMPR…

    好文分享 2025年12月17日
    000
  • C# DataSet性能最佳实践

    c# 性能优化细节 1、使用ItemArray实现对DataRow的批量赋值 在对DataRow的所有字段赋值时,使用字段名进行逐列赋值效率较低。这时应尽量使用批量字段赋值。可以使用ItemArray或rows.Add方法: / ds是数据集(DataSet)对象DataTable dt = ds.…

    好文分享 2025年12月17日
    000
  • C# 字符串操作–减少垃圾回收压力

    c# 性能优化细节 1、使用string.Empty给一个空字符串变量赋初始值 String.Empty是一个指代,而””是具体的实现 string filter=“”;//不建议 string filter=string.Empty; //建议 2、使用str.Length == 0做空串比较 最…

    好文分享 2025年12月17日
    000
  • C# 异常处理(Catch Throw)IL分析

    1、catch throw的几种形式及性能影响: private void Form1_Click(object sender, EventArgs e) { try { } catch { throw; } } private void Form1_Load(object sender, Even…

    2025年12月17日
    000
  • C# SQLite数据库 访问封装类

    在客户端配置文件节点下,添加: 其中【localdb】是本地SQLite数据库的名称,【config/local.db】是在当前程序运行目录下SQLite数据库位置 C# SQLite数据库  访问封装类代码: /// /// 本类为SQLite数据库帮助静态类,使用时只需直接调用即可,无需实例化 …

    好文分享 2025年12月17日
    000
  • C# DataRow 比较

    /// /// dataRow比较 /// /// /// /// 需要比较的列名称 /// public static bool DataRowCompare(DataRow drA, DataRow drB, string[] columnNames) { bool flag = false; …

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(5) 方法的重载

        老师都有讲课这个方法,一个老师先是在西部偏远山区,是站在教室里木头的黑板前讲课;过了几年表现好,调到了稍微好点的城市里,是坐在教室前用多媒体设备讲课;又过了几年考博士了,毕业后继续当老师,不过现在是躺在家里对着电脑远程授课。都是讲课这个方法,不同的条件下(参数不同)有不同的执行过程和输出结果…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(6) 抽象类和抽象方法

    在实际项目中,当我们设计一个父类时,经常会遇到这个类不能确定它的具体执行流程的。比如我设计一个文件类: public class AFile { private string name = string.Empty; private string path = string.Empty; priva…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(7) 方法的隐藏

    继承和抽象类中提到过,子类与父类的方法间有这些关系:子类直接使用父类方法(但是必须父类方法是public或protected类型);子类的方法覆盖父类方法(override);子类的方法重载父类方法(overload);看下面这种情况: public class YSchool { private …

    2025年12月17日
    000
  • C#基础知识整理:基础知识(8) 接口

        前面接触了抽象类,它的特点是子类必须实现abstract修饰的方法。以及还有virtual修饰的方法,virtual修饰的方法子类可以重写也可以不重写而直接使用。但是由于c#也是单继承的,所以定义一个父类,然后继承来扩展一些类的时候,会遇到不合适的情况。因为我们程序员是用程序语言来描述世界的…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(9) 接口的应用

        前面接触过接口的概念,其实接口就是一组方法、属性,然后谁继承了它,谁就要实现这组方法和属性。也就是说这个类具备了这个接口定义的一些能力。    接口的这种特性在平常的程序编写中是有很大作用的,往往完成一个大项目需要很多人同时完成,这样难免会有一些类都需要某些方法,而且执行流程都差不多。这是在…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(10) 静态

        如果想访问某个类的方法或属性,一定要先实例化该类,然后用该类的对象加.号访问。比如:有一个用户类和一个处理密码(加密和解密)的类。没生成一个用户实例后,处理密码类要对密码进行加密和解密。 using System;using System.Collections.Generic;using …

    2025年12月17日
    000
  • C#基础知识整理:基础知识(11) 值类型,引用类型

    c#是面向对象的语言,在面向对象的思想中,只有对象,所有事物都可以用类描述。所以比如这些,int,bool,char,string,double,long等都是类,那么像,30,2.5,”test”都是对应类的一个对象。 static void Main(string[] a…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(12) 超类Object

        面向对象三大特性:封装,继承,多态。那么类是从哪里继承呢?在面向对象语言中有基类或者也叫做超类的概念,也就是所有类都是从这个类继承得来的,这个超类叫object。.net中是这样描述object类的:    支持 .net framework 类层次结构中的所有类,并为派生类提供低级别服务。…

    2025年12月17日
    000
  • C#基础知识整理:基础知识(13) 异常

        往往我们在写代码的时候,总会在运行中遇到某些问题而导致程序崩溃。这并不是编程人员的水平不行,而是由业务逻辑,操作系统,或者电脑等其它设备出现问题而造成,比如在c#中经常用到user32.dll里的一些方法,假如这个文件被删掉了,你的程序照样运行不了。当然作为一个有水平的程序员总会在写程序时是…

    好文分享 2025年12月17日
    000
  • C#基础知识整理:基础知识(14) 数组

    无论哪种语言,肯定会有集合的概念。而最简单,最直观的集合应该就是数组了,数组是在内存中连续的一段空间。看看c#中数组 的定义。1、int[] intArry ; intArry= new int[6]; 这里声明了一个int数组类型变量intArry,并保存一个具有6个单元的int数组对象; int…

    2025年12月17日
    000

发表回复

登录后才能评论
关注微信