当前位置: 首页 > 知识库问答 >
问题:

为什么Swift 3需要@转义注释?

毋修为
2023-03-14

我读了这篇问答和Cocoacasts博文,我完全理解什么是< code>@escaping注释。

但老实说,我不明白我们为什么需要它。

上述Cocoacasts博客文章指出:

默认情况下,使闭包不转义有几个好处。最明显的好处是性能和编译器优化代码的能力。如果编译器知道闭包是非转义的,它可以处理内存管理的许多细节。

但是,Swift编译器可以确定<code>转义

这也意味着您可以在非转义闭包中使用self关键字而不会出现问题,因为闭包是在函数返回之前调用的。在闭包中不需要使用对self的弱引用。这是一个很好的好处,你可以免费得到。

但是,如果闭包参数被标记为@escaping我仍然可以传递使用对 self 的强引用的闭包,并且编译器不会显示任何警告。实际上,如果默认情况下@escaping闭包捕获的所有引用都是的,并且应用特殊关键字使它们变强,那将更有用。

我还认为,< code>@escaping注释是通过显式声明这个闭包参数不会对函数体进行转义来使代码自文档化的方法,但是调用方这样做的目的是什么呢?它不限制闭包定义的方式,也不阻止调用方使用强引用。所以我只希望调用方仔细查看函数签名,并采取适当的措施(比如使用弱引用)。

因此,问题是为什么我们真的需要Swift 3中的@转义闭包,以及在哪些情况下我们离不开它?

更新:

我知道不转义闭包不能传递给函数,该函数的闭包参数标记为@escaping

func testNoEscape(f: () -> ()) {
    f()
}

var storeF: (() -> ())?

func testEscaping(f: @escaping () -> ()) {
    storeF = f
}

func tryPassNoEscapeToEscaping(f: () -> ()) {
    testEscaping(f: f)
}

导致编译错误:

passing non-escaping parameter 'f' to function expecting an @escaping closure

但这是@escaping关闭带来的唯一真正的限制,它看起来像是围绕自己建立的,并没有带来任何其他好处。

更新2

虽然我正确地表达了上述想法,但我的最后一个问题不准确。

真正的问题是,如果编译器可以自己检测转义闭包,并且< code>@escaping注释不对参数值施加任何限制,那么我们为什么还需要< code>@escaping注释?

在我看来,如果编译器不允许我们在逃避闭包(如使用self和其他强引用)时做一些坏事,那将更有用。或者,如果转义闭包是某种特殊类型,那么在调用具有转义闭包参数的函数时,我们必须提到这一点:

func f(c: () -> ()) { // c is escaping from f somehow
    // ...
}

f escaping { // have to use `escaping` keyword 
    // ...
}

因此,调用方不必查看 f 签名即可知道 c 正在转义,因为如果它尝试将非转义闭包作为转义闭包参数值传递,它将出现编译错误。

在当前的实现中,希望在其代码中使用f的开发人员必须查看f,以了解c将逃逸,这是不安全的,因为这要求最初编写此代码并在以后修改它的任何人都必须详细了解f签名,这不可靠,并且此类代码没有自我记录。

我明白也许我的问题不适合这样。很抱歉。

如果是这样的话,如果我不能从在Swift语言和编译器中实现转义逻辑的人那里得到答案,我将稍后关闭它。

共有3个答案

桑鸿志
2023-03-14

编译器可以查看您创建的代码,但无法查看您用作预编译框架一部分的代码(截至目前,由于ABI问题,没有Swift框架,但将来会有)。

也许他们可以将这种转义需求仅应用于“公共”函数,但这看起来会有点不一致。最好只是通过工具提示插入关键字。

华萧迟
2023-03-14

也许还有其他原因,但最主要的原因是要传达您的闭包将超过您将其传入的函数的持续时间。这可能会对程序的行为产生非常复杂的影响,因此明确这一点很重要。

此外,闭包的“逃逸性”是公共API的一部分。编译器无法查看正在调用的编译库。如果没有@escaping属性公开传递闭包无法转义的事实,编译器无法从编译后的代码中自行推断出闭包。

督劲
2023-03-14

非转义闭包可以做转义闭门无法做的事情。它们是完全不同的动物。

例如,非转义闭包可以引用self的属性,而无需明确表示self。这是因为,作为非转义(即它在收到后立即执行),它没有以某种棘手的方式捕获self并导致保留循环的危险。

而且,非转义闭包可以在<code>inout</code>参数上关闭。但这对于逃避关闭毫无意义,是不允许的。

如果一个闭包似乎有时需要< code>self,有时不需要< code>self,有时允许对< code>inout进行闭包,而有时不需要,这将是非常神秘的。< code>@escaping注释使得这种规则区分清晰一致。

 类似资料:
  • 互联网是超文本标记语言(HTML)页面的集合,它们彼此链接以形成概念性信息网络。随着时间的推移,静态资源数量增加,图像等更丰富的项目开始成为Web结构的一部分。 高级服务器技术允许动态服务器页面 - 其内容基于查询生成的页面。 很快,需要拥有更多动态网页才能获得动态超文本标记语言(DHTML)。一切都归功于JavaScript。在接下来的几年中,我们看到了跨帧通信,试图避免页面重新加载,然后在帧内

  • 当前信息时代,哪里都是应用程序。这些应用程序们不仅仅是运行人们工作场所的工具 - 它们现在正在经营人们的生活。 对即时响应的需求,完美的行为和更多的功能是前所未有的。 而且,当然,人们期望应用程序在不同类型的设备上运行平稳,特别是在移动设备上。 应用程序执行的速度与它所做的一样重要。 NGINX的核心功能,例如其具有高性能HTTP和反向代理服务器的大规模可扩展事件驱动架构,访问和带宽控制以及与各种

  • 开发人员和运营工程师是两个不同的组织团队,如果发现这两个团队在错误的轨道上协作,则表明需要DevOps。以下是两个团队经常出现的一些问题: 在DevOps之前,开发和运营团队完全孤立。 测试和部署是在设计构建之后完成的独立活动。因此,他们比实际构建周期消耗更多时间。 在不使用DevOps的情况下,团队成员将大量时间花在测试,部署和设计上,而不是构建项目。 手动代码部署会导致生产中出现人为错误 编码

  • 问题内容: 根据PreparedStatement.setNull的Java文档,“注意:您必须指定参数的SQL类型”。该方法需要列的SQL类型的原因是什么? 我注意到传递java.sql.Types.VARCHAR还可用于非varchar列。是否存在不适合使用VARCHAR的方案(某些列类型或某些数据库提供程序)? 谢谢。 问题答案: 根据PreparedStatement.setNull的Ja

  • 问题内容: 我知道的是,全局变量和静态变量存储在段中,而未初始化的数据存储在段中。我不明白的是,为什么我们有专用于未初始化变量的段?如果未初始化的变量在运行时分配了值,那么该变量是否仅仍存在于段中? 在以下程序中, 在段中,并且在段中;那是对的吗?如果我的理解是错误的,请纠正我。 另外,请考虑以下程序, 问题答案: 原因是减小程序大小。想象一下,您的C程序在嵌入式系统上运行,其中代码和所有常量都保

  • 问题内容: 为什么需要放入GUI更新代码? 为什么Swing本身无法在内部对其进行处理?为什么调用者必须关心swing如何处理UI更新? 问题答案: 摆动对象不是线程安全的。顾名思义,允许在以后的某个时间执行任务;但更重要的是,该任务将在AWT事件分配线程上执行。使用时r,任务是异步执行的;还有,直到任务完成执行后才会返回。