为什么在 .Net 4.5 中更改了内存初始化规则



更新我的一些代码示例以进行训练时,我注意到以前由 CLR 保持未初始化的某些本地内存现在已清零

下面是一个显示"问题"的小型 CIL 示例:

.assembly Test{}
.assembly extern mscorlib{}
.class S extends [mscorlib]System.ValueType
{
    .field public int32 n;
}
.method static void F()
{
    .locals (int32, valuetype S)
    ldloc 0
    call void [mscorlib]System.Console::WriteLine(int32)
    ldloca 1
    ldfld int32 S::n
    call void [mscorlib]System.Console::WriteLine(int32)
    ret
}
.method static void Main()
{
    .entrypoint
    .locals (int32, valuetype S)
    ldloc 0
    call void [mscorlib]System.Console::WriteLine(int32)
    ldloca 1
    ldfld int32 S::n
    call void [mscorlib]System.Console::WriteLine(int32)
    call void F()
    ret
}

以下是输出:

  • .Net 2.0(我几乎可以肯定它与.Net 4.0相同(:

    $ /Windows/Microsoft.NET/Framework/v2.0.50727/ilasm.exe Test.il
    Microsoft (R) .NET Framework IL Assembler.  Version 2.0.50727.5420
    Copyright (c) Microsoft Corporation.  All rights reserved.
    Assembling 'Test.il'  to EXE --> 'Test.exe'
    Source file is ANSI
    Assembled global method F
    Assembled global method Main
    Creating PE file
    Emitting classes:
    Class 1:        S
    Emitting fields and methods:
    Global  Methods: 2;
    Class 1 Fields: 1;
    Resolving local member refs: 3 -> 3 defs, 0 refs, 0 unresolved
    Emitting events and properties:
    Global
    Class 1
    Resolving local member refs: 0 -> 0 defs, 0 refs, 0 unresolved
    Writing PE file
    Operation completed successfully
    Test.il(6) : warning -- Non-sealed value class, made sealed
    $ ./Test.exe
    211384
    0
    2157704
    2157328
    
  • .Net 4.5:

    $ /Windows/Microsoft.NET/Framework/v4.0.30319/ilasm.exe Test.il
    Microsoft (R) .NET Framework IL Assembler.  Version 4.0.30319.17929
    Copyright (c) Microsoft Corporation.  All rights reserved.
    Assembling 'Test.il'  to EXE --> 'Test.exe'
    Source file is ANSI
    Assembled global method F
    Assembled global method Main
    Creating PE file
    Emitting classes:
    Class 1:        S
    Emitting fields and methods:
    Global  Methods: 2;
    Class 1 Fields: 1;
    Resolving local member refs: 3 -> 3 defs, 0 refs, 0 unresolved
    Emitting events and properties:
    Global
    Class 1
    Resolving local member refs: 0 -> 0 defs, 0 refs, 0 unresolved
    Writing PE file
    Operation completed successfully
    Test.il(6) : warning : Non-sealed value class, made sealed
    $ ./Test.exe
    0
    0
    0
    0
    

我很好奇这种实现更改背后的基本原理。

因此,欢迎任何反馈,

尤其是来自内部人士的反馈,或对任何文章的引用,或者更好的是,对规范的引用。 :)

提前谢谢。

这只是一个巧合,没有什么能故意清除你的当地人。抖动试图将本地变量保留在寄存器中,如果没有必要,它甚至不会分配堆栈空间。你的程序很小,当你不使用/32bitpreferred 标志编译它时,它很有可能为你本地选择一个寄存器,该寄存器以前在当前线程上没有使用过,并且仍然具有零值。在以下情况下,您可以取回垃圾:

    使用/32 位
  • 首选标志编译代码。 32 位代码的寄存器较少,因此您的局部变量可能存储在脏堆栈或脏寄存器中。
  • 为值类型创建更多字段,或使用更多局部变量,以便它们不适合寄存器集,或者抖动不会选择将它们存储在那里。
  • 在你的方法中做更多的事情,涉及寄存器

我不确定为什么与以前的版本相比,您需要付出更多努力来查找 4.5 版中的未初始化值。也许这是抖动中某些变化的副作用,但我只是猜测。关键是你仍然需要局部变量上的 init 标志,以确保它们获得零初始值。

最新更新