暂无图片
暂无图片
暂无图片
暂无图片
暂无图片

C#中的托管与非托管,内存泄漏和内存溢出,你真的知道吗?

927

在.net 编程环境中,系统的资源分为托管资源非托管资源。 

  字面理解托管,就是托付个别人管理,要的是结果,具体怎么完成的我并不关心,就像某些'牛逼'的老板“我只要结果”那样。 

  在.NET FRAMEWORK中,托管给CLR(公共语言运行时)有类型安全检查,内存管理,垃圾回收等。 

  对于托管的资源的内存,你不用直接操作内存,你需要的时候跟我说。我替你申请,然后给你用,你用完可以告诉我,我帮你释放,如果你忙,忘记告诉我了,我也会在定期去帮你释放的。和你打交道的不是直接的内存,而是.NET CLR。

  托管的资源的回收工作,是不需要人工干预的,而且你也无法干预他们的回收(有GC Garbage Collection),所能够做的只是了解.net CLR如何做这些操作。也就是说对于应用程序创建的大多数对象,可以依靠 .NET Framework 的垃圾回收器隐式地执行所有必要的内存管理任务。 

 

  非托管,就是自己是又是员工又是老板,什么都做,累的要死,亲力亲为。 

  非托管的意思就是你要自己负责管理内存,这里所说的内存管理。实际上只是堆(heap)上的内存管理,栈(stack)内存和以前的一样,函数退出则释放,heap(堆)上的内存,非托管内存需要自己分配,调用构造函数(new),使用完毕后,需要自己释放这个内存,如果你不小心,把内存的指针弄丢了,就造成内存泄露了,这个内存泄露在你程序退出之前是无法弥补的。

  对于非托管资源,您在应用程序中使用完这些非托管资源之后,必须显示的释放他们,例如System.IO.StreamReader的一个文件对象,必须显示的调用对象的Close()方法关闭它,否则会占用系统的内存和资源,而且可能会出现意想不到的错误。 

总结区分托管资源和非托管资源

(1)托管资源一般是指被CLR控制的内存资源,这些资源的管理可以由CLR来控制,例如程序中分配的对象,作用域内的变量等。 

(2)非托管资源是CLR不能控制或者管理的部分,这些资源有很多,比如文件流,数据库的连接,系统的窗口句柄,打印机资源 等,这些资源一般情况下不存在于Heap(内存中用于存储对象实例的地方)中。

1、C#几乎所有对象都为托管对象,不同点是有的对象封装了非托管资源。

2、C#大部分对象在进行垃圾回收时都可以回收,包括非托管资源,因为非托管资源都已经通过C#类进行了封装,会将非托管资源的释放放在析构函数中,同时会实现IDipose接口。

3、IDipose作用是可以通过using手动提前释放,节约宝贵的资源。

4、在C#中,凡是继承了IDisposable接口的类,都可以使用using语句,从而在超出作用域后,让系统自动调用Dispose()方法。一个资源安全的类,都实现了IDisposable接口和析构函数。提供手动释放资源和系统自动释放资源的双保险。

5、类似FileStream,会同时实现IDipose,及定义~FileStream(),~FileStream()作用是使用者没有调用IDipose,在GC回收时再回收非托管资源,否则非托管资源只有在程序停止之后才会释放。

析构函数只能由垃圾回收器调用。Despose()方法只能由类的使用者调用。


.NET的优势之一就是内置了垃圾回收器GC,它帮助我们实现了自动化内存管理。但是GC再好,也有老马失前蹄的时候,它不能保证提供一个解决内存泄漏的万无一失的解决方案。

内存泄漏的危害

  • 长时间运行,程序变卡,性能严重下降

  • 程序莫名其妙挂掉

  • OutOfMemoryError错误

  • 乱七八糟的错误,还不易排查


  • 内存泄漏(memory leak ) 内存溢出 (out of memory)

  • 内存泄露 :是指程序在申请内存后,无法释放已申请的内存空间就造成了内存泄漏,一次内存泄漏似乎不会有大的影响,但内存泄漏堆积后的后果就是内存溢出。

  • 我们知道了内存泄漏的原因而内存溢出则有可能是因为我们我们多次内存泄漏堆积后的后果则变成了内存溢出

  • 内存溢出:指程序申请内存时,没有足够的内存供申请者使用,或者说,给了你一块存储int类型数据的存储空间,但是你却存储long类型的数据,那么结果就是内存不够用,此时就会报错OOM,即所谓的内存溢出,简单来说就是自己所需要使用的空间比我们拥有的内存大内存不够使用所造成的内存溢出。



文章转载自跟着阿笨一起玩NET,如果涉嫌侵权,请发送邮件至:contact@modb.pro进行举报,并提供相关证据,一经查实,墨天轮将立刻删除相关内容。

评论