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

Sye*_*yed 5 .net c# clr garbage-collection

MyObject obj = new MyObject();
Run Code Online (Sandbox Code Playgroud)

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

我的问题是CLR如何做到以下.

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

Sho*_*ikh 2

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

\n\n

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

\n\n

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

\n\n
    \n
  1. 全局/静态对象引用
  2. \n
  3. CPU寄存器
  4. \n
  5. 对象终结引用(稍后详细介绍)
  6. \n
  7. 互操作引用(传递给 COM/API 调用的 .NET 对象)
  8. \n
  9. 堆栈引用(这里主要使用)
  10. \n
\n\n

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

\n\n

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

\n\n

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

\n\n

任何与这 5 个根无关的引用都容易发生 GC。

\n\n

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

\n\n

一个简单的例子可以是:

\n\n
class MyClass \n{ \n    string Test="Hello world Wazzup!"; \n    byte[] data=new byte[86000];  \n}\n
Run Code Online (Sandbox Code Playgroud)\n\n

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

\n\n

\xe2\x80\xa2 19 个字符

\n\n

\xe2\x80\xa2 86,000 字节。

\n\n

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

\n\n

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

\n\n

问候。

\n