你知道.NET框架下的自动内存管理吗?

  C#使用的自动内存管理,使用开发者从繁重的手工分配、释放内存的操作解放出来。内存的自动管理是由垃圾回收器来执行。一个对象使用内存的生命周期是这样的:  当对象被创建时,它便分配了一定的内存,当构造器中的代码开始运行时,这个对象就“活”了。

  当这个对象或者是它的任何一部分在可以预计的将来已经没有任何作用时,这个对象将不会再使用,它就应当被销毁。

  一旦这个对象符合了对销毁的条件,在一定的时间后,这个对象的销毁器就将被执行,一般情况下,除非被显示地重写,这个销毁器只能运行一次。

  一旦销毁器被运行,那么这个对象以及它任何一部分都不可能在以后的运行中使用,这甚至包括正在运行的销毁器。这时这个对象将被认为是不可见的,它所占的资源将会被回收。

  最后由垃圾回收器释放这个对象所占的资源。

  垃圾回收器控制着这些对象的使用信息并利用这些信息控制内存,比如内存哪里创建了一个新对象,什么时候重新创建对象以及什么时候将这个对象释放。

  像其它的语言一样,C#假定确实存在这样一个垃圾回收器,而且这个垃圾回收器可以管理很大范围的内存。比如,C#并不要求销毁器一事实上要执行,也不要求对象一旦无用则马上回收。

  当然垃圾回收器的行为也是可以被控制的,这个控制的方法来自于System.GC类。这个类可以请求回收、销毁器运行等操作。

  下面是一个例子。

  

      class A

  {

  ~A() {

  Console.WriteLine("Destruct instance of A");

  }

  }

  class B

  {

  object ref;

  public B(object o) {

  ref = o;

  }

  ~B() {

  Console.WriteLine("Destruct instance of B");

  }

  }

  class TMest

  {

  static void Main() {

  B b = new B(new A());

  b = null;

  GC.Collect();

  GC.WaitForPendingFinalizers();

  }

  }

  上面的程序创建了类A与类B的一个实例,当变量b被赋于null值时,A与B均符合了垃圾回收器的回收要求。此时就没有任何代码能够访问它们了。

  执行的结果,下面两种情况都有可能:

  Destruct instance of A

  Destruct instance of B

  与

  Destruct instance of B

  Destruct instance of A

  因为上面的程序的并没有限制这两个对象被回收的顺序。

  在某些敏感的条件下,有关区分“销毁”与“回收”操作条件的定义是非常重要的:

  

      class A

  {

  ~A() {

  Console.WriteLine("Destruct instance of A");

  }

  public void F() {

  Console.WriteLine("A.F");

  TMest.RefA = this;

  }

  }

  class B

  {

  public A Ref;

  ~B() {

  Console.WriteLine("Destruct instance of B");

  Ref.F();

  }

  }

  class TMest

  {

  public static A RefA;

  public static B RefB;

  static void Main() {

  RefB = new B();

  RefA = new A();

  RefB.Ref = RefA;

  RefB = null;

  RefA = null;

  // A and B now eligible for destruction

  GC.Collect();

  GC.WaitForPendingFinalizers();

  // B now eligible for collection, but A is not

  if (RefA != null)

  Console.WriteLine("RefA is not null");

  }

  }

  在上面的程序中,如果垃圾回收器选择先执行类B的销毁器,那么执行的结果为:

  Destruct instance of A

  Destruct instance of B

  A.F

  RefA is not null

  注意,虽然实例A并没有被使用,但是从输出的结果大家可以看到A的销毁器确实执行了,而且连A的方法F也被执行了。同时我们也注意至,一个对象销毁器的运行又可能使一个实例变得可用。在这种情况下,实例B销毁器的执行使得先前并没有调用的实例A也可以被访问了,而这一种就是引用RefA的功劳,当调用WaitForPendingFinalizers方法以后,实例B就可以被垃圾回收器回收,而此时的实例A则还不可以。

  为了区分这些行为,大家编写程序时,最好只管理当前类的销毁器,而不要采用引用其它类的实例或者静态字段。

时间: 2024-10-28 12:07:18

你知道.NET框架下的自动内存管理吗?的相关文章

C# 语言规范--1.4 自动内存管理

规范 手动内存管理要求开发人员管理内存块的分配和回收.手动内存管理可能既耗时又麻烦.在 C# 中提供了自动内存管理,使开发人员从这个繁重的任务中解脱出来.在绝大多数情况下,自动内存管理可以提高代码质量和开发人员的工作效率,并且不会对表达能力或性能造成负面影响. 示例 using System; public class Stack {    private Node first = null;    public bool Empty {       get {          return

Oracle 11g的Memory_target与自动内存管理

Oracle11g的自动内存管理(Automatic Memory Management)的新特性是Oracle在内存管理上的又一重要增强.如果这个参数设置过高,在实例启动时可能会出现如下错误提示: SQL*Plus: Release 11.1.0.5.0 - Beta on Sun Jul 29 08:35:28 2007 Copyright (c) 1982, 2007, Oracle. All rights reserved. Connected to an idle instance.

Oracle 11g的自动内存管理

Oracle11g的自动内存管理(Automatic Memory Management)的新特性是Oracle在内存管理上的又一重要增强.如果这个参数设置过高,在实例启动时可能会出现如下错误提示: SQL*Plus: Release 11.1.0.5.0 - Beta on Sun Jul 29 08:35:28 2007 Copyright (c) 1982, 2007, Oracle. All rights reserved. Connected to an idle instance.

[CLR via C#]21. 自动内存管理(垃圾回收机制)

原文:[CLR via C#]21. 自动内存管理(垃圾回收机制) 目录 理解垃圾回收平台的基本工作原理 垃圾回收算法 垃圾回收与调试 使用终结操作来释放本地资源 对托管资源使用终结操作 是什么导致Finalize方法被调用 终结操作揭秘 Dispose模式:强制对象清理资源 使用实现了Dispose模式的类型 C#的using语句 手动监视和控制对象的生存期 对象复活 代 线程劫持 大对象 一.理解垃圾回收平台的基本工作原理 值类型(含所有枚举类型).集合类型.String.Attribute

Java虚拟机自动内存管理

生活规律告诉我们,在享受便利的同时一般都会付出巨大的代价,如果你在享受了便利的同时,还没有为此付出代价,不是说明没有,只是还没到付出的时候.试问,有哪个Java系统架构师不懂Java虚拟机?纵观Java程序员的发展历程,又有多少人是卡在了Java虚拟机之上.所以如果你还没有感觉到为此付出代价,说明你已经Java虚拟机的糖衣炮弹所击中,且被毒害之深.Java的自动内存管理就是这样,像毒药一样,一旦上瘾就很难戒掉,而且会沉迷于此.而正确的做法就是了解其原理,拿到尚方宝剑,当虚拟机不好好为你提供服务时

Java自动内存管理详解

最近找了两本Java虚拟机方面的书,看了看其中对于Java自动内存管理的章节,写的都大同小异,在此总结一下,主要是三个方面:内存划分.内存分配.内存回收. 内存划分(运行时数据区) JVM运行时数据区 从线程的角度来分,可分为线程私有和线程共享的,上图中左边的灰色区域就是线程共享的区域,包括堆.方法区.运行时常量池.而右边的区域则是线程私有的,包括程序计数器.虚拟机栈. 堆 堆是虚拟机管理的内存中最大的一块,是被线程共享的一块区域,主要用于存放对象实例,但并不是所有对象都是在堆上分配的.同时堆也

设置oralce自动内存管理

  设置oralce自动内存管理 启用oracle自动内存管理需要shutdown ,restart 1.确定sga pga内存大小: SHOW PARAMETER TARGET 2.确定自数据库启动以来pga最大的使用大小: select value from v$pgastat where name='maximum PGA allocated'; 3.计算memory_target大小: memory_target = sga_target + max(pga_aggregate_targ

设置oralce自动内存管理执行步骤_oracle

设置oralce自动内存管理 启用oracle自动内存管理需要shutdown ,restart 1.确定sga pga内存大小: 复制代码 代码如下: SHOW PARAMETER TARGET 2.确定自数据库启动以来pga最大的使用大小: 复制代码 代码如下: select value from v$pgastat where name='maximum PGA allocated'; 3.计算memory_target大小: 复制代码 代码如下: memory_target = sga_

Oracle 10G数据库自动内存管理分析

是不是很难准确地分配不同的池所需的内存数?自动共享内存管理特性使得自动将内存分配到最需要的地方去成为可能. 无论您是一个刚入门的 DBA 还是一个经验丰富的 DBA,您肯定至少看到过一次类似以下的错误: ORA-04031:unable to allocate 2216 bytes of shared memory ("shared pool"... ... 或者这种错误: ORA-04031:unable to allocate XXXX bytes of shared memory