- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我试图在自定义集合中实现写时复制行为,这时我遇到了捕获 self
的嵌套函数的这种奇怪行为。
在下面的代码中,调用了isKnownUniquelyReferenced(_:)
将始终返回 false,尽管在 甚至定义嵌套函数之前被调用:
class Foo {}
struct Bar {
var foo = Foo()
public mutating func bar() {
print(isKnownUniquelyReferenced(&foo))
func nestedFunc() {
_ = self // capture self
}
nestedFunc()
}
}
var b = Bar()
b.bar() // false ?!
到底为什么会这样,我能做些什么来解决它(假设 nestedFunc
实际上对 self
做了一些有用的事情)?
我知道 self
的捕获可能会干扰对 isKnownUniquelyReferenced(_:)
的调用——但肯定会在调用 时发生nestedFunc()
在这种情况下?
最佳答案
从 Xcode 8.3 beta 中可用的 Swift 3.1 开始,此问题已得到修复。 self
在方法中根本不再装箱,因此 isKnownUniquelyReferenced(_:)
返回 true
正如预期的那样。
我认为这是一个错误,并已提交错误报告 ( SR-3530 )。然而,我对这个问题的原因很感兴趣,所以进行了一些挖掘——这就是我的发现。
查看为 bar()
生成的规范 SIL方法(对于 -Onone 构建)显示 Swift 正在为 alloc_box
堆分配一个盒子( self
)在方法的最开始——这样它就可以被nestedFunc()
捕获.
// Bar.bar() -> ()
sil hidden @main.Bar.bar () -> () : $@convention(method) (@inout Bar) -> () {
// %0 // users: %10, %3
bb0(%0 : $*Bar):
// create new heap-allocated box, and store self in it.
// this is where the problem stems from – there are now two copies of the Bar instance, thus isKnownUniquelyReferenced will return false.
%1 = alloc_box $Bar, var, name "self", argno 1, loc "main.swift":15:26, scope 9 // users: %11, %9, %7, %2
%2 = project_box %1 : $@box Bar, loc "main.swift":15:26, scope 9 // users: %10, %5, %3
copy_addr %0 to [initialization] %2 : $*Bar, scope 9 // id: %3
// call isKnownUniquelyReferenced (I removed the print() function call as it generates a bunch of unrelated SIL).
// function_ref isKnownUniquelyReferenced<A where ...> (inout A) -> Bool
%4 = function_ref @Swift.isKnownUniquelyReferenced <A where A: Swift.AnyObject> (inout A) -> Swift.Bool : $@convention(thin) <τ_0_0 where τ_0_0 : AnyObject> (@inout τ_0_0) -> Bool, loc "main.swift":17:9, scope 10 // user: %6
%5 = struct_element_addr %2 : $*Bar, #Bar.foo, loc "main.swift":17:35, scope 10 // user: %6
%6 = apply %4<Foo>(%5) : $@convention(thin) <τ_0_0 where τ_0_0 : AnyObject> (@inout τ_0_0) -> Bool, loc "main.swift":17:39, scope 10
// retain the heap-allocated box containing self, in preparation for applying nestedFunc() with it.
// (as it's passed as an @owned parameter).
strong_retain %1 : $@box Bar, loc "main.swift":27:9, scope 10 // id: %7
// call the nested function with the box as the argument.
// function_ref Bar.(bar() -> ()).(nestedFunc #1)() -> ()
%8 = function_ref @main.Bar.(bar () -> ()).(nestedFunc #1) () -> () : $@convention(thin) (@owned @box Bar) -> (), loc "main.swift":27:9, scope 10 // user: %9
%9 = apply %8(%1) : $@convention(thin) (@owned @box Bar) -> (), loc "main.swift":27:20, scope 10
// once called, copy the contents of the box back to the address of the Bar instance that was passed into the method, and release the box.
copy_addr %2 to %0 : $*Bar, scope 10 // id: %10
strong_release %1 : $@box Bar, loc "main.swift":29:5, scope 10 // id: %11
// so cute.
%12 = tuple (), loc "main.swift":29:5, scope 10 // user: %13
return %12 : $(), loc "main.swift":29:5, scope 10 // id: %13
}
( Full SIL here )
由于这个装箱,现在有 两个 Bar
的副本bar()
中的实例方法,因此意味着 isKnownUniquelyReferenced(_:)
将返回 false,因为有两个对 Foo
的引用实例。
据我所知,self
的拳击在方法的开头似乎是 mutating
优化的结果从复制输入复制输出的方法(self
在方法调用开始时被装箱,然后将突变应用于该盒子,然后在方法结束时写回被调用方)到按引用传递(这种优化发生在原始 SIL 和规范 SIL 之间。
用于创建 self
副本的同一个盒子现在使用方法内的变异来捕获self
调用嵌套函数。我看不出为什么不应该在调用 nestedFunc()
之前创建捕获框,因为这是捕获 self
的合乎逻辑的位置(而不是在方法的开头)。
尽管无论如何,首先创建一个盒子是完全多余的,因为 nestedFunc()
不会也不能逃脱。试图返回 nestedFunc()
从该方法产生以下编译器错误:
Nested function cannot capture inout parameter and escape
所以它看起来真的只是一个尚未优化的极端情况。即使在 -O 构建中,尽管 Bar
的堆分配实例能够针对 foo
优化为堆栈分配属性,这仍然会导致对 Foo
的不必要的第二次引用实例。
一个解决方案是只添加一个 inout
self
参数 nestedFunc()
, 允许 self
只是通过引用传递,而不是被捕获:
func nestedFunc(_ `self`: inout Bar) {
_ = self // do something useful with self
}
// ...
nestedFunc(&self)
现在生成 SIL (-Onone):
// function_ref Bar.(bar() -> ()).(nestedFunc #1)(inout Bar) -> ()
%5 = function_ref @main.Bar.(bar () -> ()).(nestedFunc #1) (inout main.Bar) -> () : $@convention(thin) (@inout Bar) -> (), loc "main.swift":31:9, scope 10 // user: %6
%6 = apply %5(%0) : $@convention(thin) (@inout Bar) -> (), loc "main.swift":31:25, scope 10
此解决方案的优势在于它只是一种直接的按引用传递(因为 Bar
参数标记为 @inout
)。因此,Bar
的实例只存在一个副本。 – 因此 isKnownUniquelyReferenced(_:)
可以返回true。
另一种可能的解决方案,如果self
未在 nestedFunc()
内发生突变, 就是通过 self
按值,而不是引用。这可以通过本地闭包中的捕获列表来完成:
let nestedFunc = { [`self` = self] in // copy self into the closure.
_ = self // the self inside the closure is immutable.
}
// ...
nestedFunc()
优点是你不需要显式传递任何东西到 nestedFunc()
称呼。因为 Bar
的实例在创建闭包之前不会按值传递——它不会干扰对 isKnownUniquelyReferenced(_:)
的调用,假设调用先于闭包创建。
关于swift - 为什么嵌套的自捕获函数会干扰 isKnownUniquelyReferenced(_ :)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41449125/
尝试使用集成到 QTCreator 的表单编辑器,但即使我将插件放入 QtCreator.app/Contents/MacOS/designer 也不会显示。不过,相同的 dylib 文件确实适用于独
在此代码示例中。 “this.method2();”之后会读到什么?在返回returnedValue之前会跳转到method2()吗? public int method1(int returnedV
我的项目有通过gradle配置的依赖项。我想添加以下依赖项: compile group: 'org.restlet.jse', name: 'org.restlet.ext.apispark', v
我将把我们基于 Windows 的客户管理软件移植到基于 Web 的软件。我发现 polymer 可能是一种选择。 但是,对于我们的使用,我们找不到 polymer 组件具有表格 View 、下拉菜单
我的项目文件夹 Project 中有一个文件夹,比如 ED 文件夹,当我在 Eclipse 中指定在哪里查找我写入的文件时 File file = new File("ED/text.txt"); e
这是奇怪的事情,这个有效: $('#box').css({"backgroundPosition": "0px 250px"}); 但这不起作用,它只是不改变位置: $('#box').animate
这个问题在这里已经有了答案: Why does OR 0 round numbers in Javascript? (3 个答案) 关闭 5 年前。 Mozilla JavaScript Guide
这个问题在这里已经有了答案: Is the function strcmpi in the C standard libary of ISO? (3 个答案) 关闭 8 年前。 我有一个问题,为什么
我目前使用的是共享主机方案,我不确定它使用的是哪个版本的 MySQL,但它似乎不支持 DATETIMEOFFSET 类型。 是否存在支持 DATETIMEOFFSET 的 MySQL 版本?或者有计划
研究 Seam 3,我发现 Seam Solder 允许将 @Named 注释应用于包 - 在这种情况下,该包中的所有 bean 都将自动命名,就好像它们符合条件一样@Named 他们自己。我没有看到
我知道 .append 偶尔会增加数组的容量并形成数组的新副本,但 .removeLast 会逆转这种情况并减少容量通过复制到一个新的更小的数组来改变数组? 最佳答案 否(或者至少如果是,则它是一个错
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
noexcept 函数说明符是否旨在 boost 性能,因为生成的对象中可能没有记录异常的代码,因此应尽可能将其添加到函数声明和定义中?我首先想到了可调用对象的包装器,其中 noexcept 可能会产
我正在使用 Angularjs 1.3.7,刚刚发现 Promise.all 在成功响应后不会更新 angularjs View ,而 $q.all 会。由于 Promises 包含在 native
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我正在编写一个玩具(物理)矢量库,并且遇到了 GHC 坚持认为函数应该具有 Integer 的问题。是他们的类型。我希望向量乘以向量以及标量(仅使用 * ),虽然这可以通过仅使用 Vector 来实现
PHP 的 mail() 函数发送邮件正常,但 Swiftmailer 的 Swift_MailTransport 不起作用! 这有效: mail('user@example.com', 'test
我尝试通过 php 脚本转储我的数据,但没有命令行。所以我用 this script 创建了我的 .sql 文件然后我尝试使用我的脚本: $link = mysql_connect($host, $u
使用 python 2.6.4 中的 sqlite3 标准库,以下查询在 sqlite3 命令行上运行良好: select segmentid, node_t, start, number,title
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我是一名优秀的程序员,十分优秀!