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

规范

手动内存管理要求开发人员管理内存块的分配和回收。手动内存管理可能既耗时又麻烦。在 C# 中提供了自动内存管理,使开发人员从这个繁重的任务中解脱出来。在绝大多数情况下,自动内存管理可以提高代码质量和开发人员的工作效率,并且不会对表达能力或性能造成负面影响。

示例

using System;

public class Stack

{

   private Node first = null;

   public bool Empty {

      get {

         return (first == null);

      }

   }

   public object Pop() {

      if (first == null)

         throw new Exception("Can't Pop from an empty Stack.");

      else {

         object temp = first.Value;

         first = first.Next;

         return temp;

      }

   }

   public void Push(object o) {

      first = new Node(o, first);

   }

   private class Node

   {

      public Node Next;

      public object Value;

      public Node(object value): this(value, null) {}

      public Node(object value, Node next) {

         Next = next;

         Value = value;

      }

   }

}

显示了一个 Stack 类,它实际上是 Node 实例的一个链接表。Node 实例是在 Push 方法中创建的,当不再需要它们时会对其进行垃圾回收。当任何代码都不再可能访问某个 Node 实例时,该实例就成为垃圾回收的对象。例如,当从 Stack 中移除某项时,相关的 Node 实例就符合了垃圾回收条件,等待被回收。

示例

class Test

{

   static void Main() {

      Stack s = new Stack();

      for (int i = 0; i < 10; i++)

         s.Push(i);

      s = null;

   }

}

显示了使用 Stack 类的代码。该代码创建了一个 Stack 类,并用 10 个元素初始化该类,然后给它赋值 null。给变量 s 赋了 null 值后,Stack 及关联的 10 个 Node 实例变得符合垃圾回收条件。垃圾回收器可以立即清除上述实例,但并没要求它一定做到立即清除。

为 C# 服务的基础垃圾回收器可以通过在内存中移动对象来工作,而这种移动对于大多数 C# 开发人员是不可见的。对于那些通常满足于自动内存管理、但有时又需要精确控制或细微性能调整的开发人员,C# 提供了编写“不安全”代码的能力。这类代码可以直接处理指针类型和对象地址。但是,C# 要求程序员固定对象,暂时阻止垃圾回收器移动它们。

从开发人员和用户的角度,这种“不安全”代码功能实际上是一种“安全”的功能。在代码中必须用修饰符 unsafe 清楚地标记出不安全代码,以便开发人员不可能不经意地使用不安全语言功能,并且编译器和执行引擎协同工作,确保不安全代码无法假冒安全代码。这些限制将不安全代码的使用仅限于代码受到信任的情况。

示例

using System;

class Test

{

   static void WriteLocations(byte[] arr) {

      unsafe {

         fixed (byte* pArray = arr) {

            byte* pElem = pArray;

            for (int i = 0; i < arr.Length; i++) {

               byte value = *pElem;

               Console.WriteLine("arr[{0}] at 0x{1:X} is {2}",

                  i, (uint)pElem, value);

               pElem++;

            }

         }

      }

   }

   static void Main() {

      byte[] arr = new byte[] {1, 2, 3, 4, 5};

      WriteLocations(arr);

   }

}

显示了一个名为 WriteLocations 的方法。它含有一个不安全块,该块固定了一个数组实例,然后使用指针操作实现逐个地访问该数组的元素。每个数组元素的索引、值和位置被写入控制台。下面是一个可能的输出示例:

arr[0] at 0x8E0360 is 1

arr[1] at 0x8E0361 is 2

arr[2] at 0x8E0362 is 3

arr[3] at 0x8E0363 is 4

arr[4] at 0x8E0364 is 5

当然,确切的内存位置可能因应用程序的不同执行而异。

时间: 2024-12-22 10:01:29

C# 语言规范--1.4 自动内存管理的相关文章

Java虚拟机自动内存管理

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

Java自动内存管理详解

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

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

C#使用的自动内存管理,使用开发者从繁重的手工分配.释放内存的操作解放出来.内存的自动管理是由垃圾回收器来执行.一个对象使用内存的生命周期是这样的: 当对象被创建时,它便分配了一定的内存,当构造器中的代码开始运行时,这个对象就"活"了. 当这个对象或者是它的任何一部分在可以预计的将来已经没有任何作用时,这个对象将不会再使用,它就应当被销毁. 一旦这个对象符合了对销毁的条件,在一定的时间后,这个对象的销毁器就将被执行,一般情况下,除非被显示地重写,这个销毁器只能运行一次. 一旦销毁器被运

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.

设置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

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

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

设置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