对Swift中some和any关键字的理解

简介: 在最新Swift版本中(Xcode14,Swift5.7),如果协议中有使用泛型,则如果要将此协议作为参数类型,必须使用any关键字进行修饰。其实在Swift5.1中也引入过一个some关键字,any和some都适用于协议,这两个关键字从语义上和写法上对泛型的使用进行了优化。

对Swift中some和any关键字的理解

在最新Swift版本中(Xcode14,Swift5.7),如果协议中有使用泛型,则如果要将此协议作为参数类型,必须使用any关键字进行修饰。其实在Swift5.1中也引入过一个some关键字,any和some都适用于协议,这两个关键字从语义上和写法上对泛型的使用进行了优化。

1. any

我们知道,协议中会规定一些属性和方法,用来约束其他结构的实现。举个简单的例子,我们可以使用协议定义了一个可飞行的实例需要实现的方法和属性,如下:

protocol Fly {
    var name:String {get set}
    func fly()
}

class Bird:Fly {
    var name = "Bird"
    func fly() {
        print(name + "Fly")
    }
}

func test(f: Fly) {
    f.fly()
}

test(f: Bird())

这个程序当前运行的很好,语义也很明确,即test的函数的参数需要是实现了Fly协议的任意类型,其实在此中情况下,虽然在调用是我们传入的是Bird实例,但是由于协议类型的约束较弱,在函数执行时编译器会将其解释成了Fly类型,实际上产生了类型丢失。尤其是当协议中有使用泛型时,此时上面的写法在最新的Xcode版本中会提示错误,需要我们添加any关键字。any关键字的意义其实就是实现上述的语义,将参数类型定义为遵守某个协议的任意类型,如下:

import Foundation

protocol Fly {
    associatedtype T
    var name:T {get set}
    func fly()
}


class Bird:Fly {
    typealias T = String
    var name = "Bird"
    func fly() {
        print(name + "Fly")
    }
}

func test(f: any Fly) {
    f.fly()
}

let f = test(f: Bird())

2.some

针对于上面代码的应用场景,我们只需要约束参数的类型是遵守Fly协议的即可,但是有时候这并不够,有时协议中的函数会需要多个参数,我们需要使用泛型约束其参数的类型一致,例如:

import Foundation

protocol Fly {
    associatedtype T
    var name:T {get set}
    func fly()
    func add(a:T, b:T)
}


class Bird:Fly {
    typealias T = String
    var name = "Bird"
    func fly() {
        print(name + "Fly")
    }
    func add(a: String, b: String) {
        
    }
}

func test(f: any Fly) {
    f.fly()
    // 这里会报错 因为any Fly类型在运行时无法确定成某个具体的类型
    f.add(a: f.name, b: f.name)
}

test(f: Bird())

可以看到,上面的代码中,test函数会报错,核心的原因在于any Fly类型的语音是任意实现了Fly协议的类型,无论是编译时还是运行时,编译器都无法推导出此f参数的类型。要解决上面的问题,可以采用泛型的方式来改写,如下:

func test<T:Fly>(f: T) {
    f.fly()
    f.add(a: f.name, b: f.name)
}

此时代码则没有任何问题了,some关键字其实也是用于这一种场景,其表示的是一种透明类型,在运行时编译器知道其具体的类型是什么,只是对调用方来说是抽象的。下面的写法与上面使用泛型的写法作用完全一致:

func test(f: some Fly) {
    f.fly()
    f.add(a: f.name, b: f.name)
}

整体看来,相对与泛型那种写法,使用some的写法语义更加清晰,风格上也与any刚好一致。

      最后,我们再来总结下,整体看来,any和some都是用来描述语义的关键字,any和协议一起使用,表示的是语义比较传统,及遵守了某个协议的类型,具体什么类型编译器也不知道。而some和协议一起使用表示的是具象的一个类型,此类型编译时不知道,调用时也开发者来说也是透明的,但是编译器自己是知道的,它就是具体的一个类型。

目录
相关文章
|
Swift
Swift中AnyObject、Any、AnyClass、T.self、T.Type、type(of:)、Self的使用和区别
Swift中AnyObject、Any、AnyClass、T.self、T.Type、type(of:)、Self的使用和区别
213 0
|
存储 编译器 测试技术
Swift 之关键字总结
在Swift官方文档的词汇结构中, 有非常多的关键字, 它们被用于声明中、语句中、表达式中、类中、模式中, 还有以数字符号#开头的关键字, 以及特定上下文环境使用的关键字。 本文中涉及的代码可以在这里下载代码资源。
523 0
|
存储 Swift
Swift5.1—Any 和 AnyObject 的类型转换
Swift5.1—Any 和 AnyObject 的类型转换
289 0
12.Swift学习之Any、AnyObject与类型转化
Any、AnyObject Any是一个空协议集合的别名,它表示没有实现任何协议,因此它可以是任何类型,包括类实例与结构体实例。可以表示任何类型,包括函数类型。
1958 0
|
数据安全/隐私保护 iOS开发
[译] Swift + 关键字(V 3.0.1)
本文讲的是[译] Swift + 关键字(V 3.0.1),有句话以前说过,现在我要再次提一下,一个优秀的匠人,他(她)的工具同样优秀。当我们一丝不苟地去使用这些工具时,它们就会带我们到想去的地方,或者完成我们的梦寐以求的作品。
1383 0
《从零开始学Swift》学习笔记(Day 47)——final关键字
<span style="font-family:宋体;font-size: 10.5pt; mso-ascii-font-family: Arial; mso-fareast-font-family: 宋体; mso-fareast-theme-font: minor-fareast; mso-hansi-font-family: Arial; mso-bidi-font-family: A
1255 0
《从零开始学Swift》学习笔记(Day5)——我所知道的标识符和关键字
<div style="top: 0px;"></div> <span style="font-family:宋体;font-size:14px;"></span><p style="margin: 0cm 0cm 0pt; text-indent: 20pt;"><span style="font-size:14px;"><strong style="mso-bidi-font-weig
1746 0