.Net7自定义GC垃圾回收器

net7,自定义,gc,垃圾,回收 · 浏览次数 : 65

小编点评

**自定义GC垃圾回收器的步骤:** 1. **创建一个C++ DLL库项目**,其中包含两个头文件:`gcenv.base.h`和`gcinterface.h`。 2. **实现三个接口:** * `IGCHandleManagerIGCHandleStoreIGCHeap`:用于管理GC堆中的内存和指针。 * `IGCHeap`:用于分配和释放内存。 * `GC_VersionInfo`:用于获取GC版本信息。 3. **导出两个函数:** * `IGCHandleManager* GC_CreateInstance()`: 用于创建GC句柄。 * `void GC_DestroyInstance(IGCHandleManager* handle)`:用于释放GC句柄。 4. **修改`GC_VersionInfo`函数**,设置要显示的版本信息。 5. **编写自定义GC分配和释放函数**,将它们与`GC_CreateInstance`和`GC_DestroyInstance`函数关联。 6. **在项目属性中设置环境变量`DOTNET_GCName`**,指定自定义GC的名称。 7. **运行代码**,查看版本信息,应该显示自定义的版本号。 **示例代码:** ```cpp #include "gcenv.base.h" #include "gcinterface.h" // ... void GC_VersionInfo(VersionInfo* result) { result->MajorVersion = GC_INTERFACE_MAJOR_VERSION; result->MinorVersion = GC_INTERFACE_MINOR_VERSION; result->BuildVersion = 0; result->Name = "Custom"; } // ... // 在GC_CreateInstance函数中创建GC句柄 IGCHandleManager* GC_CreateInstance() { // 创建GC句柄 } // 在GC_DestroyInstance函数中释放GC句柄 void GC_DestroyInstance(IGCHandleManager* handle) { // 释放GC句柄 } ```

正文

1.前言

CLR和GC高度耦合,.Net7里面分离CLR和GC,则比较容易实现这件事情。本篇来看下,自定义一个GC垃圾回收器。

2.概述

这里首先演示下自定义GC垃圾回收后的效果。
1.下载Custom.dll
2.找到当前.Net目录,比如这里的7.0.10

C:\Program Files\dotnet\shared\Microsoft.NETCore.App\7.0.10

把Custom.dll复制到此目录
3.比如有以下C#代码:

 static void Main(string[] args){
     Console.WriteLine("Hello Main");
     Program pm = new Program();
     pm = null;
     GC.Collect();
 }

在项目属性-》调试-》常规-》打开调试启动配置文件UI-》环境变量里面填写如下键值:

名称:DOTNET_GCName
值:Custom.dll

4.运行此段代码,显示如下
image
hello Jianghupt这三条语句就是Custom.dll里面自定义了GC的标识效果。表示它是自定义的。
5.原理
CLR在初始化的时候,会判断是否有DOTNET_GCName环境变量,如果有则替换掉默认的GC回收器,用DOTNET_GCName指定的GC回收器。这里很明显的就是进入了自定义垃圾回收器里面去了。如果取消掉DOTNET_GCName环境变量,则不会有这三句话。可以自己去尝试下。


如何自定义GC垃圾回收呢?

自定义GC垃圾回收器,需要进行如下操作。新建一个C++ DLL库项目
引入两个头文件,实现三个接口

#include "gcenv.base.h"
#include "gcinterface.h"

IGCHandleManager
IGCHandleStore
IGCHeap

需要导出两个函数

extern "C" DLLEXPORT HRESULT
GC_Initialize(
    /* In */  IGCToCLR* clrToGC,
    /* Out */ IGCHeap** gcHeap,
    /* Out */ IGCHandleManager** gcHandleManager,
    /* Out */ GcDacVars* gcDacVars
)
{
    printf("Hello Jianghupt! This Is Custom GC Into CLR!!!!!!!!!\r\n");
    printf("Hello Jianghupt! Welcom To Customer GC CLR!!!!!!!!!\r\n");
    printf("Hello Jianghupt! Microsfot The GC Update Customer!!!!!!!!!\r\n");
    IGCHeap* heap = new CustomGCHeap(clrToGC);
    IGCHandleManager* handleManager = new CustomGCHandleManager();
    *gcHeap = heap;
    *gcHandleManager = handleManager;
    return S_OK;
}

extern "C" DLLEXPORT void
GC_VersionInfo(
    /* Out */ VersionInfo* result
)
{
	result->MajorVersion = GC_INTERFACE_MAJOR_VERSION;
	result->MinorVersion = GC_INTERFACE_MINOR_VERSION;
    result->BuildVersion = 0;
    result->Name = "Custom";
}

可以看到GC_Initialize里面的三句话,就是上面控制台显示的三句话。这个时候它实际上已经进入到了自定义GC里面去了。

源码下载

有任何问题,可以关注公众号(jianghupt),了解更多。
image

与.Net7自定义GC垃圾回收器相似的内容:

.Net7自定义GC垃圾回收器

1.前言 CLR和GC高度耦合,.Net7里面分离CLR和GC,则比较容易实现这件事情。本篇来看下,自定义一个GC垃圾回收器。 2.概述 这里首先演示下自定义GC垃圾回收后的效果。 1.下载Custom.dll 2.找到当前.Net目录,比如这里的7.0.10 C:\Program Files\do

.NET周报【12月第4期 2022-12-31】

祝大家新年快乐! 国内文章 『 再看.NET7』数值类型 https://mp.weixin.qq.com/s/ctiBMPY6Hditk81AzHSRng 在C#中,有int16,用short来定义;有int32,用int定义;用int64,用long来定义。在.NET7中,添加了int128,和

.NET周报【10月最后一期 2022-11-01】

精选要闻 .NET 7 NativeAOT比.NET单文件发布文件小80% https://twitter.com/JamesNK/status/1584919726861737984?s=20&t=cOsB41s2cydu_Ibts4xnEw AOT GRPC服务器应用程序比.NET运行时自包含的

一个.NET 7 + DDD + CQRS +React+Vite的实战项目

## 项目简介 基于SignalR实现聊天通信,支持横向扩展,可支撑上万用户同时在线聊天 ## 快速体验 http://server.tokengo.top:8888/ 可在这里快速体验使用,请注意目前只适配了PC端,请勿使用手机访问,可能出现样式不适应的情况, 当然如果你想要自己部署也可以,目前提

在 Net7.0环境下测试了 Assembly.Load、Assmebly.LoadFile和Assembly.LoadFrom的区别

一、简介 很长时间没有关注一些C#技术细节了,主要在研究微服务、容器、云原生、容器编排等高大上的主题了,最近在写一些框架的时候,遇到了一些和在 Net Framework 框架下不一样的情况,当然了,我今天主要测试的是,在通过【添加项目引用】和【手动拷贝DLL】的情况下,这三个方法加载程序集:Ass

【.NET 8】ASP.NET Core计划 - 支持更完善的AOT发布

.NET7.0刚发布不久,.NET社区开始了.NET8.0的开发,重心重新回到了新功能的迭代。 我们知道在.NET7.0中一个令人激动的特新就是支持了NativeAOT,我们可以通过NativeAOT生成本机程序,由于无需JIT编译,所以无需安装.NET Runtime,也进一步的提升了.程序的启动

.NET性能系列文章一:.NET7的性能改进

这些方法在.NET7中变得更快 照片来自 CHUTTERSNAP 的 Unsplash 欢迎阅读.NET性能系列的第一章。这一系列的特点是对.NET世界中许多不同的主题进行研究、比较性能。正如标题所说的那样,本章节在于.NET7中的性能改进。你将看到哪种方法是实现特定功能最快的方法,以及大量的技巧和

试试将.NET7编译为WASM并在Docker上运行

之前有听到说Docker支持Wasmtime了,刚好.NET7也支持WASM,就带大家来了解一下这个东西,顺便试试它怎么样。 因为WASM(WebAssembly) 一开始是一个给浏览器的技术,比起JS解释执行,WASM能用于提升浏览器的用户体验,因为在一些场景中它有着比JS更好的性能。 大家可以将

C#11之原始字符串

最近.NET7.0和C#11相继发布,笔者也是第一时间就用上了C#11,其中C#11的有一个更新能解决困扰我多年的问题,也就是文章的标题原始字符串。 在使用C#11的原始字符串时,发现的一些有意思的东西,超出了我原本对它的期待,话不多说,我们一起来看看。 多年的困扰 我不知道大家有没有写过这样的代码

使用.NET7和C#11打造最快的序列化程序-以MemoryPack为例

## 译者注 本文是一篇不可多得的好文,MemoryPack 的作者 neuecc 大佬通过本文解释了他是如何将序列化程序性能提升到极致的;其中从很多方面(可变长度、字符串、集合等)解释了一些性能优化的技巧,值得每一个开发人员学习,特别是框架的开发人员的学习,一定能让大家获益匪浅。 ## 简介 我发