从C++看C#托管内存与非托管内存

news/2024/11/15 0:55:29/文章来源:https://www.cnblogs.com/ggtc/p/18333486

进程的内存

一个exe文件,在没有运行时,其磁盘存储空间格式为函数代码段+全局变量段。加载为内存后,其进程内存模式增加为函数代码段+全局变量段+函数调用栈+堆区。我们重点讨论堆区。

进程内存
函数代码段
全局变量段
函数调用栈
堆区

托管堆与非托管堆

  • C#
    int a=10这种代码申请的内存空间位于函数调用栈区

    var stu=new Student();
    GC.Collect();
    

    new运算符申请的内存空间位于堆区。关键在于new关键字。在C#中,这个关键字是向CLR虚拟机申请空间,因此这个内存空间位于托管堆上面,如果没有对这个对象的引用,在我们调用GC.Collect()后,或者CLR主动收集垃圾,申请的这段内存空间就会被CLR释放。这种机制简化了内存管理,我们不能直接控制内存的释放时机。不能精确指定释放哪个对象占用的空间。

    我不太清楚CLR具体原理,但CLR也只是运行在操作系统上的一个程序。假设它是C++写的,那么我们可以想象,CLR调用C++new关键字后向操作系统申请了一个堆区空间,然后把这个变量放在一个全局列表里面。然后记录我们运行在CLR上面的C#托管程序堆这个对象的引用。当没有引用存在之后,CLR从列表中删除这个对象,并调用delete xxx把内存释放给操作系统。

    但是非托管堆呢?

  • C++
    在C++中也有new关键字,比如

    Student* stu=new Student();
    delete stu;
    //引发异常
    cout >> stu->Name >> stu->Age;
    

    申请的内存空间也位于堆区。但又C++没有虚拟机,所以C++中的new关键字实际上是向操作系统申请内存空间,在进程关闭后,又操作系统释放。但是C++给了另一个关键字deletedelete stu可以手动释放向操作系统申请的内存空间。之后访问这个结构体的字段会抛出异常

  • C
    C语言中没有new关键字,但却有两个函数,mallocfree

    int* ptr = (int *)malloc(5 * sizeof(int));
    free(ptr);
    

    他们起到了和C++中new关键字相同的作用。也是向操作系统申请一块在堆区的内存空间。

C#通过new关键字向CLR申请的内存空间位于托管堆。C++通过new关键字向操作系统申请的内存空间位于非托管堆。C语言通过mallocfree向操作系统申请的内存空间也位于非托管堆。C#的new关键字更像是对C++的new关键字的封装。

C#如何申请位于非托管堆的内存空间

C#本身的new运算符申请的是托管堆的内存空间,要申请非托管堆内存空间,目前我知道的只有通过调用C++的动态链接库实现。在.net8以前,使用DLLImport特性在函数声明上面。在.net8,使用LiberyImport特性在函数声明上面

C++部分

新建一个C++动态链接库项目
image

然后添加.h头文件和.cpp源文件

//Student.h#pragma once
#include <string>
using namespace std;extern struct Student
{wchar_t* Name;// 使用 char* 替代 std::string 以保证与C#兼容int Age;
};//__declspec(xxx)是MSC编译器支持的关键字,dllexport表示导出后面的函数
/// <summary>
/// 创建学生
/// </summary>
/// <param name="name">姓名</param>
/// <returns>学生内存地址</returns>
extern "C" __declspec(dllexport) Student* CreateStudent(const wchar_t* name);/// <summary>
/// 释放堆上的内存
/// </summary>
/// <param name="student">学生地址</param>
extern "C" __declspec(dllexport) void FreeStudent(Student* student);
//Student.cpp//pch.h在项目属性中指定,pch.cpp必需
#include "pch.h"#include "Student.h"
#include <cstring>Student* CreateStudent(const wchar_t* name)
{//new申请堆空间Student* student = new Student;student->Age = 10;//new申请名字所需要的堆空间//wcslen应对unicode,ansi的话,使用strlen和char就够了student->Name = new wchar_t[wcslen(name) + 1];//内存赋值wcscpy_s(student->Name, wcslen(name) + 1, name);return student;
}void FreeStudent(Student* student)
{// 假设使用 new 分配delete[] student->Name;//释放数组形式的堆内存delete student; 
}

生成项目后,在解决方案下的x64\Debug中可以找到DLL

C#部分

由于C++动态链接库不符合C#动态链接库的规范。所以没法在C#项目的依赖中直接添加对类库的引用。只需要把DLL放在项目根目录下,把文件复制方式改为总是复制,然后代码中导入。

[DllImport("Student.dll", //指定DLL
CharSet=CharSet.Unicode//指定字符串编码
)]
public static extern IntPtr CreateStudent(string name);[DllImport("Student.dll")]
private static extern IntPtr FreeStudent(IntPtr stu);public static void Main()
{string studentName = "John";//用IntPtr接收C++申请空间的起始地址IntPtr studentPtr = CreateStudent(studentName);// 在C#中操作Student结构体需要进行手动的内存管理,如下// 从地址所在内存构建C#对象或结构体,类似于指针的解引用Student student = Marshal.PtrToStructure<Student>(studentPtr);// 访问学生信息//Marshal.PtrToStringUni(student.Name)将一段内存解释为unicode字符串,直到遇见结束符'\0'Console.WriteLine($"Student Name: {Marshal.PtrToStringUni(student.Name)}, Age: {student.Age}");// 记得释放分配的内存FreeStudent(studentPtr);
}// 定义C++的Student结构体
[StructLayout(LayoutKind.Sequential)]
public struct Student
{// IntPtr对应C++中的 char*public IntPtr Name;public int Age;
}

调用结果如下

image

非托管类释放非托管内存空间

如果我们把C++代码的调用封装成类,那么可以实现IDisposable接口。在Dispose方法中释放资源,然后使用using语句块来确保Dispose方法被调用。这样使得内存泄漏可能性降低。

继承IDisposable接口后按下alt+enter,选择通过释放模式实现接口可以快速生成代码

/// <summary>
/// 非托管类
/// </summary>
public class Student:IDisposable
{// 定义C++的Student结构体[StructLayout(LayoutKind.Sequential)]private struct _Student{public IntPtr Name;public int Age;}// IntPtr对应C++中的 char*//需要在Dispose中手动释放private IntPtr _this;private IntPtr name;public string Name => Marshal.PtrToStringUni(name);public int Age;private bool disposedValue;public Student(string name){_this=CreateStudent(name);_Student layout = Marshal.PtrToStructure<_Student>(_this);//记住要释放的内存起始地址this.Age = layout.Age;this.name = layout.Name;}[DllImport("Student.dll", CharSet = CharSet.Unicode)]private static extern IntPtr CreateStudent(string name);[DllImport("Student.dll")]private static extern IntPtr FreeStudent(IntPtr stu);protected virtual void Dispose(bool disposing){if (!disposedValue){if (disposing){// TODO: 释放托管状态(托管对象)}// TODO: 释放未托管的资源(未托管的对象)并重写终结器if (_this != IntPtr.Zero){FreeStudent(_this);//设置为不可访问_this = IntPtr.Zero;name = IntPtr.Zero;}// TODO: 将大型字段设置为 nulldisposedValue = true;}}// // TODO: 仅当“Dispose(bool disposing)”拥有用于释放未托管资源的代码时才替代终结器// ~Student()// {//     // 不要更改此代码。请将清理代码放入“Dispose(bool disposing)”方法中//     Dispose(disposing: false);// }public void Dispose(){// 不要更改此代码。请将清理代码放入“Dispose(bool disposing)”方法中Dispose(disposing: true);GC.SuppressFinalize(this);}
}

然后在Main中创建对象

string studentName = "John";
using (Student stu=new Student(studentName))
{Console.WriteLine($"Student Name: {stu.Name}, Age: {stu.Age}");
}
return;

结果

image

代码确实执行到了这里。

  • 单步调试执行流程,using->Console->Dispose()->Dispose(bool disposing)->FreeStudent(_this);

image

事实上可以在FreeStudent(_this);之后加一句代码Console.WriteLine(Name);,你将会看到原本的正常属性变成了乱码

image

其实代码有点重复。如果我把_Student layout = Marshal.PtrToStructure<_Student>(_this);中的layout定义为Student的私有成员,那么Student中的那两个私有指针就不需要了,完全可以从layout中取得。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/775827.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Magic-PDF:端到端PDF文档解析神器 构建高质量RAG必备!

项目结构流程解析 预处理的作用是判断文档内容是否需要进行OCR识别,如果是普通可编辑的PDF文档,则使用PyMuPDF库提取元信息。 模型层除了常规的OCR、版面结构分析外,还有公式检测模型,可提取公式内容,用于后续把公式转化为Latex格式。但是目前暂无表格内容识别,官方预计1…

Windows系统常用端口详解

135端口135主要用于Microsoft的远程过程调用(RPC)服务。RPCSS(Remote Procedure Call Subsystem) 服务是 COM 和 DCOM 服务器的服务控制管理器。它执行 COM 和 DCOM 服务器的对象激活请求、对象导出程序解析和分布式垃圾回收。如果此服务被停用或禁用,则使用 COM 或 DCOM 的…

Albumentations库使用

介绍Albumentations的核心使用方法,提供对应测试代码1 Albumentations库介绍 一个好用的开源图像处理库,适用于对RGB、灰度图、多光谱图像,以及对应的mask、边界框和关键点同时变换。通常用于数据增广,是PyTorch生态系统的一部分。 主页:https://albumentations.ai/ 2 核心…

LinkAI RAG知识库平台优化之路

LinkAI RAG知识库平台支持无结构文档、Q&A问答对、多列表格以及网站内容自动导入,并加入了自研的增强解析功能支持对文档中图片以及表格的自动解析。支持基于语义的向量检索和基于关键词的全文检索的增强混合检索功能,生成的回复可以标注答案来源,同时可以在使用记录中查…

14. 迭代器、生成器、模块与包、json模块

1.迭代器 1.1 迭代器介绍 迭代器是用来迭代取值的工具 每一次迭代得到的结果会作为下一次迭代的初始值,单纯的重复并不是迭代# while循环实现迭代取值 a = [1, 2, 3, 4, 5, 6] index = 0 while index < len(a):print(a[index])index += 1 1.2 可迭代对象 内置有_ _iter_ _方…

# 代码随想录二刷(哈希表)

代码随想录二刷(哈希表) 三数之和思路反正对于我来说是真的难想出来。若这道题还是采用哈希表的思路去做,非常麻烦,并且还要考虑去重的操作。所以这道题其实用双指针,是更方便的。具体程序如下: class Solution:def threeSum(self, nums: List[int]) -> List[List[int]]…

ctfshow-web入门-nodejs系列

web334 下载源码后缀改为zip打开即可 先对源码经行一个简单的分析 login.js// 引入Express框架 var express = require(express);// 创建一个路由实例 var router = express.Router();// 引入用户数据,假设user模块导出的是一个包含用户项的对象 var users = require(../modul…

2021年我因为Tab Session Manager丢失数据,好像是研究过一次leveldb的查看/解码方式 但是当时好像因为时间关系没能成功 / chrome .ldb文件

Default\Local Storage\leveldb .ldb2023年下半年我因为chatmindai修改域名,又研究过一次,因为时间关系也没有细究最近,我想查看一下anki的devtool的Local Storage,即https://ankiweb.net/shared/info/31746032这个插件产生的 C:\Users\xxx\AppData\Local\Anki\QtWebEngine…

联合省选 2024 Day2T1 迷宫守卫 题解

联合省选 2024 Day2T1 迷宫守卫 题解 好像距离联合省选已经半年了,前两天看到这题才想起来改,距离分班已经半年了,也算是好好学了半年了,但是还是那么菜,有点绷不住,感觉不如文化课 后来翻到题解区第二篇题解才知道自己赛时想的反悔贪心其实是正解,但是当时啥也不会,主…

小白程序员也要对世界进行第一次的呐喊!

身为程序员对世界的第一声呐喊——Hello World!新建一个文件夹 新建一个目录,并将其命名为Hello.java(关键一步)注意!文件类型显示的是java文件才成功 (文件的后缀要改为java)双击文件,开始编写(本人使用的是Notepad++进行编写) 输入图片中的代码(全部要用英文输入法…

Django模板、模版语言和静态文件

1. templates模板(html)在app目录下创建一个templates目录,用于存放网页模板利用url返回网页点击查看代码 def user_list(request):return render(request,"user_list.html");输入url地址时,会去app目录下的templates目录下寻找名为user_list的HTML文件(根据app…