Swift 中 Opaque Types学习指南
<div id="navCategory"><h5 class="catalogue">目录</h5><ul class="first_class_ul"><li>什么是Opaque Types</li><ul class="second_class_ul"><li>如何使用</li></ul><li>为什么要使用</li><ul class="second_class_ul"><li>可读性更强</li><li>性能更好</li><li>使用的限制</li></ul><li>总结</li><ul class="second_class_ul"></ul></ul></div><p class="maodian"></p><h2>什么是Opaque Types</h2><div class="cros igoods"><div class="goodsin" data-img="https://img14.360buyimg.com/pop/jfs/t3229/39/2924280771/392779/d242f52a/57e8d68eN57129bbf.jpg" data-name="精通Swift设计模式(图灵出品)" data-owner="京东自营" data-price="77.4" data-tgid="38" data-url="https://union-click.jd.com/jdc?e=&p=JF8BAMoJK1olXwUFXVxUDU8eC18IG1MTWgIAVm4ZVxNJXF9RXh5UHw0cSgYYXBcIWDoXSQVJQwYCXFhaDEkVHDZNRwYlOkAYXB1bXFV0SBEKRiZjCAAEUS0ZaEcbM2gNHF4dXwMBZF5eDkwXAmoIK2sVXDZQOobrvpOysnPcsdTA1ZEyVW5dD0IWCm0PGVISXA4FZF5VDHtUVypcWBhdbTYyV25tOEsnAF9WdVpGWwVQVgwPZhZBBydVGRJJMwQHXVpcCE0RBV8KGloXXzYy"></div></div>
<p>Opaque Types 是在 Swift5.7 新添加的一个特性。主要是和 Protocol 来搭配使用,用来定义函数中的参数类型。它的作用就是隐藏参数的具体类型,从而使代码变得更加通用,减少冗余代码。废话不多说,下面来看一看它如何使用。</p>
<p class="maodian"></p><h3>如何使用</h3>
<p>假设我们有一个定义动物行为的 Protocol Behavior,有两个动物类 Cat 和 Dog 遵守了该协议:</p>
<div class="jb51code"><pre class="brush:cpp;">protocol Behavior {
func run()
}
struct Cat: Behavior {
func run() {
print("Cat run")
}
}
struct Dog: Behavior {
func run() {
print("Dog run")
}
}
</pre></div>
<p>如果我们想测试一下 run 的功能,可以通过使用 Opaque Types 来定义函数的参数,从而使测试函数可以接受 Cat 和 Dog 的实例对象:</p>
<div class="jb51code"><pre class="brush:cpp;">func testRun(animal: some Behavior) {
animal.run()
}
let cat = Cat()
let dog = Dog()
testRun(animal: cat) // Cat run
testRun(animal: dog) // Dog run
</pre></div>
<p>从上述代码可以看到,animal 的类型为 <code>some Behavior</code>。通过 some 关键字声明的类型就是所谓的 Opaque Types,<code>some Behavior</code> 的意思就是任何遵守了 Behavior 的类的实例对象。</p>
<p>看到这里,可能有的读者就会发现这个不是什么新奇的特性啊,这不就相当于泛型嘛?有这种想法的读者稍安勿躁,下面就来讲一下为什么要使用 Opaque Types。</p>
<p class="maodian"></p><h2>为什么要使用</h2>
<p class="maodian"></p><h3>可读性更强</h3>
<p>首先,下面的三个函数从语法层级上来说都是等价的:</p>
<div class="jb51code"><pre class="brush:cpp;">func testRun(animal: some Behavior) {
animal.run()
}
func testRun1<A: Behavior>(animal: A) {
animal.run()
}
func testRun2<A>(animal: A) where A: Behavior {
animal.run()
}
</pre></div>
<p>为什么使用的第一个原因就是:使用 some 定义更加清晰,使得代码可读性更高更容易理解。 上面的代码定义比较简单,可能感觉差别不大,大家看看下面的代码应该会有更深的体会😂:</p>
<div class="jb51code"><pre class="brush:cpp;">func encodeAnyDictionaryOfPairs(_ dict: ) -> Data
func encodeAnyDictionaryOfPairs<_T1: Hashable & Codable, _T2: Codable, _T3: Codable>(_ dict: ) -> Data
</pre></div>
<p class="maodian"></p><h3>性能更好</h3>
<p>还有一个重要的原因就是当 Opaque Types 当做返回值类型时它的限制比 Protocol 类型更加严格,比如下面的代码:</p>
<div class="jb51code"><pre class="brush:cpp;">func testReturn(isCat: Bool) -> some Behavior { // 编译报错
if isCat {
return Cat()
} else {
return Dog()
}
}
func testReturn1(isCat: Bool) -> Behavior { // 正常编译不会报错
if isCat {
return Cat()
} else {
return Dog()
}
}
</pre></div>
<p>当编译器编译 testReturn 函数时,会报以下错误:</p>
<p style="text-align:center"><img alt="" src="https://img.jbzj.com/file_images/article/202304/2023040709090905.jpg" /></p>
<p>而 testReturn1 则不会报错。因为对于 Protocol 当做返回值来讲,不会强制要求所有分支返回同一类型的对象。</p>
<p>当 Opaque Types 做返回值时,虽然返回值的具体类型不做限制,即任何遵守 Behavior 协议的对象都可以,但所有的分支返回必须为同一类型的对象,即可能返回 Cat,又可能返回 Dog 是不允许的。</p>
<p>这样做的好处就是代码的性能更好,因为我们从底层限制了返回类型必须是同一类型,所以编译器处理起来会更快。</p>
<p>任何的事物都有两面性,Opaque Types 也不例外。最后我们来了解一下它的限制。</p>
<p class="maodian"></p><h3>使用的限制</h3>
<ul><li>不能用于可变参数的声明</li><li>不用用于匿名函数的声明</li></ul>
<p class="maodian"></p><h2>总结</h2>
<p>本篇文章,首先讲解了什么是 Opaque Types,然后又了解了它的使用方式,接着学习了它的优点:可读性更高;性能更好。最后说明了一下它的限制。 希望大家看完这篇文章,能在项目中多多实践。</p>
<p>参考链接</p>
<ul><li>Apple Document</li><li>Swift evolution</li><li>understanding some and any</li></ul>
<p>更多关于Swift Opaque Types的资料请关注琼殿技术社区其它相关文章!</p>
<div class="art_xg">
<b>您可能感兴趣的文章:</b><ul><li>Swift 并发修改Sendable 闭包实例详解</li><li>Swift之for循环的基础使用学习</li><li>Swift简单快速的动态更换app图标AppIcon方法示例</li><li>Swift自动调整视图布局AutoLayout和AutoresizingMask功能详解</li><li>Swift 中的 RegexBuilder学习指南</li><li>Swift中的高阶函数功能作用示例详解</li><li>Swift并发系统并行运行多个任务使用详解</li></ul>
</div>
</div>
<!--endmain-->
頁:
[1]