假设我有三个对象:“a”、“b”和“c”。 对象“a”和“c”是长期存在的、静态引用的服务单例。 对象“b”是短暂的,即没有静态引用使它保持事件状态。
现在假设对象“a”在其方法之一的范围内创建对象“b”的实例,例如
B b = new B();
进一步假设 B 类看起来像这样:
public B ()
{
C.ActionList.Add ( SomeMethod );
}
void SomeMethod ()
{
...
}
现在,对象“b”能活多久?我的假设是它存在于调用其构造函数的方法的范围之外;具体来说,只要它的方法仍在对象“c”的“ActionList”中。
这样对吗?如果没有,并且它被垃圾收集,当“c”运行其“ActionList”中的所有方法时会发生什么?
奖励问题:如果 'b' 上的方法没有命名,而是匿名的,并按如下方式写在构造函数中怎么办:
public B ()
{
C.ActionList.Add ( () => {
...
} );
}
最佳答案
specifically, for as long as its method is still in the 'ActionList' of object 'c'.
是的,没错。实例方法的委托(delegate)创建实例本身的“硬引用”,并将使其保持事件状态。你的第二个问题不相关。
请注意,这就是为什么事件订阅是 .NET 中“内存泄漏”的常见来源 - 它们在技术上不会泄漏,但事件订阅是基于委托(delegate)的,并且具有相同的行为,因此事件订阅持有一个对实例的引用。
What if the method on 'b' is not named, but anonymous and written in a lambda in the constructor?
这个是一样的。如果 lambda 使用相关实例的状态,它将被编译器转换为一个类型的实例方法,并引用该实例,并持有一个引用。 (请注意,如果某些 lambda 表达式不依赖于任何封闭值、实例等,则有可能将其转换为静态方法,在这种情况下,它不会持有引用。)
在您的情况下,...
的内容将决定这一点。如果您的表达式只是: () => { Console.WriteLine("Foo"); }
,它不需要关闭实例中的任何值,也不会使用有问题的实例,因此它不会持有引用。如果你这样做 () => { Console.WriteLine(this.Foo); }
,但是,它会在引用 this
的类型上创建一个方法,并使类实例保持事件状态。
关于c# - 将一个对象的方法传递给另一个对象是否使第一个对象保持事件状态?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23020789/