老赵面试题参考答案(三)

Posted by Coderidea on July 26, 2020

      在.NET程序运行过程中,什么是堆,什么是栈?什么情况下会在堆(栈)上分配数据?它们有性能上的区别吗?“结构”对象可能分配在堆上吗?什么情况下会发生,有什么需要注意的吗?

  在.NET程序运行过程中,什么是堆,什么是栈?

  堆也就是托管堆(managed heap),进程初始化的时候,CLR要保留一块连续的地址空间,这个地址空间最初并没有对应的物理存储空间。这个地址空间就是托管堆。

  栈是在程序运行过程中用于保存指令,值类型变量的内存区域(一个线程对应一个栈),栈的结构和数据结构中“栈”的结构是一样的,“先进后出”。

  什么情况下会在堆(栈)上分配数据?它们有性能上的区别吗?

  值类型在栈上分配,引用类型在堆上分配。由于在栈上分配数据不受垃圾回收的控制,不存在垃圾回收的各种开销,应用程序执行的时候垃圾回收的次数也会少很多,所以栈上分配显然要比堆上分配性能上好。由于在栈上分配的变量已经包含了实例的字段所以不需要一个指针指向它。空间上的开销也较小。

  “结构”对象可能分配在堆上吗?什么情况下会发生,有什么需要注意的吗?

      这点不是太明确:值类型不作为对象在堆中分配。但在许多情况下,都需要获取对值类型的一个实例引用。例如假定要建个ArrayList来容纳Point结构。

 //声明一个值类型。 
struct Point { public int x, y;}
public sealed class Program
{
public static void Main()
{
ArrayList a
= new ArrayList();
Point p;
for (int i = 0; i < 10; i++)
{
p.x
= p.y = i; a.Add(p);
}
}
}

   这个时候 Point 值类型必须转换成一个真正 的,在堆中托管的对象,而且必须 获取对这个对象的引用。 这就是装箱的机制。 (不知道老赵说的是不是这种情况,还望老赵予以解答,还希望知道的朋友探讨下是不是这种情况)

      下面是装箱的过程:

       1.在托管堆中分配好内存。分配的内存是值 类型的各个字段需要的内存量加上托管的所有对象都 有的两个额外成员(类型对象指针和同步索引)需要的内存量。

       2 值类型的字段复制到新分配的内存

       3 返回对象的地址。现在,这个地址是对一个对象的引用,值类型现在是一个引用类型。

      在运行时,当前存在于Point 值类型的实例P中的字段会复制到新分配的Point对象中。已经装箱的Point 对象(现在是一个引用类型)的地址会返回给Add方法。Point对象会一直存在于堆中。

     如果是这种情况那么需要注意的地方就:这样会产生装箱/ 拆箱的操作。建议用泛型集合类。 FCl现在包含一组泛型集合类。这样不需要对集合中的项进行装箱、拆箱的处理,使性能提高不少,托管堆中需要创建的对象减少了,减少了应用程序中需要执行的垃圾回收次数。

     最后一点还希望知道的朋友确认下,也希望老赵能确认下。