- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
假设我有以下代码:
public class Foo
{
private int x;
private int y;
public Bar CreateBar()
{
return new Bar(x, () => y);
}
}
[Serializable]
public class Bar
{
private int a;
private Func<int> b;
public Bar(int a, Func<int> b)
{
this.a = a;
this.b = b;
}
}
在这种情况下,对象和值的范围会发生什么变化?由于 x 是一个值类型,它按值传递给 Bar,因此,它的范围不需要发生任何事情。但是你怎么了?实际评估 b 时,需要保留 y 的值以返回。是否所有的 Foo 都保留下来以在以后评估 y ?我只能假设 Foo 没有被 GC。
现在假设我们将 Bar 序列化到磁盘,然后再反序列化它。实际连载的是什么?它也序列化了 Foo 吗?在 Bar 被反序列化后,发生了什么魔法,可以对 b 求值?您能解释一下 IL 中发生了什么吗?
最佳答案
更新:无需借助 IL 即可查看实际发生的情况:Using reflector to understand anonymous methods and captured variables
当您使用时:
public Bar CreateBar()
{
return new Bar(x, () => y);
}
你隐含的意思是this.y
;所以就委托(delegate)而言,它是对 Foo
的引用包括在内。因此,Bar
的实例(通过委托(delegate))保留整个 Foo
存活(未被垃圾收集)直到 Bar
可供收藏。
特别是,编译器不需要(在这种情况下)生成额外的类来处理捕获的变量;唯一需要的是 Foo
实例,因此可以在 Foo
上生成一个方法.如果委托(delegate)涉及局部变量(this
除外),这将更加复杂。
就序列化而言...好吧,我首先要说的是,序列化委托(delegate)是一个非常非常糟糕的主意。然而,BinaryFormatter
将走委托(delegate),你可以(理论上)结束序列化 Bar
, 一个连载 Foo
,以及一个链接它们的序列化委托(delegate) - 但仅如果您标记 Foo
作为[Serializable]
.
但我强调 - 这是一个坏主意。我很少用 BinaryFormatter
(出于各种原因),但我看到使用它的人的一个常见问题是“为什么要尝试序列化(某种随机类型)”。通常,答案是“您正在发布一个事件,它正在尝试序列化订阅者”,在这种情况下,最常见的解决方法是将事件的字段标记为 [NonSerialized]
。 .
而不是看 IL;另一种调查方法是在 .NET 1.0 模式下使用反射器(即不在匿名方法中交换);然后你可以看到:
public Bar CreateBar()
{
return new Bar(this.x, new Func<int>(this.<CreateBar>b__0));
}
[CompilerGenerated]
private int <CreateBar>b__0()
{
return this.y;
}
如你所见;东西传给了Bar
是当前实例 ( <CreateBar>b__0()
) 上隐藏方法(称为 this
)的委托(delegate)。所以它是当前Foo
的实例传递给 Bar
.
关于c# - 匿名方法、范围和序列化,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1279101/
我不能解决这个问题。和标题说的差不多…… 如果其他两个范围/列中有“否”,我如何获得范围或列的平均值? 换句话说,我想计算 A 列的平均值,并且我有两列询问是/否问题(B 列和 C 列)。我只希望 B
我知道 python 2to3 将所有 xrange 更改为 range 我没有发现任何问题。我的问题是关于它如何将 range(...) 更改为 list(range(...)) :它是愚蠢的,只是
我有一个 Primefaces JSF 项目,并且我的 Bean 注释有以下内容: @Named("reportTabBean") @SessionScoped public class Report
在 rails3 中,我在模型中制作了相同的范围。例如 class Common ?" , at) } end 我想将公共(public)范围拆分为 lib 中的模块。所以我试试这个。 module
我需要在另一个 View 范围 bean 中使用保存在 View 范围 bean 中的一些数据。 @ManagedBean @ViewScoped public class Attivita impl
为什么下面的代码输出4?谁能给我推荐一篇好文章来深入学习 javascript 范围。 这段代码返回4,但我不明白为什么? (function f(){ return f(); functio
我有一个与此结构类似的脚本 $(function(){ var someVariable; function doSomething(){ //here } $('#som
我刚刚开始学习 Jquery,但这些示例对我帮助不大...... 现在,以下代码发生的情况是,我有 4 个表单,我使用每个表单的链接在它们之间进行切换。但我不知道如何在第一个函数中获取变量“postO
为什么当我这样做时: function Dog(){ this.firstName = 'scrappy'; } Dog.firstName 未定义? 但是我可以这样做: Dog.firstNa
我想打印文本文件 text.txt 的选定部分,其中包含: tickme 1.1(no.3) lesson1-bases lesson2-advancedfurther para:using the
我正在编写一些 JavaScript 代码。我对这个关键字有点困惑。如何在 dataReceivedHandler 函数中访问 logger 变量? MyClass: { logger: nu
我有这个代码: Public Sub test() Dim Tgt As Range Set Tgt = Range("A1") End Sub 我想更改当前为“A1”的 Tgt 的引
我正忙于此工作,以为我会把它放在我们那里。 该数字必须是最多3个单位和最多5个小数位的数字,等等。 有效的 999.99999 99.9 9 0.99999 0 无效的 -0.1 999.123456
覆盖代码时: @Override public void open(ExecutionContext executionContext) { super.open(executio
我想使用 preg_match 来匹配数字 1 - 21。我如何使用 preg_match 来做到这一点?如果数字大于 21,我不想匹配任何东西。 example preg_match('([0-9]
根据docs range函数有四种形式: (range) 0 - 无穷大 (range end) 0 - 结束 (range start end)开始 - 结束 (range start end st
我知道有一个UISlider,但是有人已经制作了RangeSlider(用两个拇指吗?)或者知道如何扩展 uislider? 最佳答案 我认为你不能直接扩展 UISlider,你可能需要扩展 UICo
我正在尝试将范围转换为列表。 nums = [] for x in range (9000, 9004): nums.append(x) print nums 输出 [9000] [9
请注意:此问题是由于在运行我的修饰方法时使用了GraphQL解析器。这意味着this的范围为undefined。但是,该问题的基础知识对于装饰者遇到问题的任何人都是有用的。 这是我想使用的基本装饰器(
我正在尝试创建一个工具来从网页上抓取信息(是的,我有权限)。 到目前为止,我一直在使用 Node.js 结合 requests 和 Cheerio 来拉取页面,然后根据 CSS 选择器查找信息。我已经
我是一名优秀的程序员,十分优秀!