为什么在这个简单的F#示例中未收集弱引用?

问题描述 投票:1回答:1
open System

let WeakReferenceExample() =
    let mutable obj = new Object();
    let weak = new WeakReference(obj);

    GC.Collect();
    Console.WriteLine("IsAlive: {0}\nobj <> null is {1}\n---", weak.IsAlive, obj <> null);

    obj <- null;
    GC.Collect();
    Console.WriteLine("IsAlive: {0}", weak.IsAlive);

WeakReferenceExample()
Console.ReadKey()

Rx In Action书籍样本中翻译而来。上面的代码在运行时提供以下输出,与在C#中编译并运行它后得到的输出不同。

IsAlive: True
obj <> null is True
---
IsAlive: True

为什么不收集弱引用?

f# garbage-collection weak-references
1个回答
2
投票

这是测量影响结果的情况之一。

根本原因是在调试过程中为了获得更好的局部效果而优化了编译器(反向?)。

写时:

let list1 = [1; 2; 3]
let list2 = [3; 4; 5]
printfn "%b" (list1 = list2)

这将每个子表达式详细说明为:

let list1 = [1; 2; 3]
let list2 = [3; 4; 5] 
let list1' = list1
let list2' = list2
let is_eq = list1'.Equals(list2')
printfn "%b" (is_eq)

现在您可以开始猜测这是怎么回事了。

 Console.WriteLine("IsAlive: {0}\nobj <> null is {1}\n---", weak.IsAlive, obj <> null);

详细说明:

    let isAlive = weak.IsAlive
    let obj' = obj
    let isNotNull = obj' <> null
    Console.WriteLine("IsAlive: {0}\nobj <> null is {1}\n---", isAlive, isNotNull);

现在执行操作:

    obj <- null;
    GC.Collect();

请注意,obj'中仍然存在引用。

因此,在GC传递期间将不会收集对象。这就是WeakReference显示的内容。

非常有趣,如果您注释掉第一个Console.Writeline,则没有子表达式的详细说明,因此也没有引用,您会得到:

IsAlive: False

或者您可以仅以Release模式构建它。

© www.soinside.com 2019 - 2024. All rights reserved.