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
为什么不收集弱引用?
这是测量影响结果的情况之一。
根本原因是在调试过程中为了获得更好的局部效果而优化了编译器(反向?)。
写时:
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模式构建它。