当我给 MyObject obj = new MyObject() 时发生了什么

本文关键字:MyObject 发生了 什么 obj new | 更新日期: 2023-09-27 18:37:20

MyObject obj = new MyObject();

我知道"new"关键字将调用构造函数并在托管堆中初始化对象。

我的问题是CLR如何执行以下操作。

  1. CLR 如何执行上述行?
  2. CLR 如何为对象分配模因?
  3. CLR 如何确定对象的大小?
  4. 如果没有空间为堆中的对象分配内存,CLR 将如何知道它?

当我给 MyObject obj = new MyObject() 时发生了什么

创建引用类型(类、委托、接口、字符串或对象)时,它将被分配到堆上。Net 有四个不同的堆:(gen0, gen1, gen2)(Small Object Heap) 和 LOH(Large Object Heap)。所有 85k 或更小的内容都放在前三个堆之一上,具体取决于创建时间(从 gen0 移动到 gen1 再到 gen2,依此类推)。大于 85k 的对象被放置在 LOH 上。LOH 永远不会被压缩,因此最终,我正在执行的类型分配最终会导致 OOM 错误,因为对象分散在该内存空间中。这些称为托管堆。

要创建对象,您需要做的就是使用 new 关键字; .NET 将负责创建、初始化对象并将其放置在右堆上,并保留任何必要的额外内存。之后,您几乎可以忘记该对象,因为当您完成它时,您不必删除它。

当您使用 new 关键字创建引用类型对象时,它被放置在堆上,其引用主要用于当前正在运行的堆栈。 还有其他可能的来源,您的对象可以用作引用:

  1. 全局/静态对象引用
  2. 中央处理器寄存器
  3. 对象终结引用(稍后详述)
  4. 互操作引用(传递给 COM/API 调用的 .NET 对象)
  5. 堆栈引用(主要用于此处)

这 5 个实际上是形成对象引用层次结构的 GC 根节点。想象一下经典的 Customer 类,它通常有一个存储 Order 类的集合。将订单添加到订单集合时,集合本身将保留对添加的订单的引用。如果 Customer 类的实例也有对它的堆栈引用。

这就是复杂对象的层次结构形成方式,这就是 GC 查看引用的方式。

例如:g.客户对象的堆栈引用 ->对订单列表对象的引用 ->引用单个订单。

任何从这 5 个根中松散引用的东西都容易出现 GC。

如何将内存分配给对象有点复杂,并且它通常会按时间增长,如指定的 MKK 一样 http://msdn.microsoft.com/en-us/magazine/cc163791.aspx

一个简单的例子可以是:

class MyClass 
{ 
    string Test="Hello world Wazzup!"; 
    byte[] data=new byte[86000];  
}

很容易假设 MyClass 在分配时的大小包括:

• 19 个字符

• 86,000 字节。

事实上,对象的大小将只包括一般的类内容,以及存储指向字符串和字节数组(类级变量)的对象指针所需的内存,然后将其分别分配到堆上。字符串将在 SOH 上分配,其对象引用由类的实例持有;字节数组将被分配到 LOH 上,因为它大于 85 KB。

SOH 可以压缩,其中不会有碎片,而 LOH 可以在内存中存在碎片。

问候。

相关文章: