swift 中的 AnyObject 和 Any

1,917 阅读2分钟

诞生

swift 作为新起步的语言,必然抛不掉一些历史遗留包袱。用过 Objective-C 的同学肯定知道有一种叫做 id 的类型。他可以表示任意类的实例,编译器不会对其类型声明的变量进行检查。在用 swift 做 app 开发时,为了能适配 Cocoa 架构,AnyObject 就诞生了。它可以代表任意 class 类型(用来替代OC中的 id)。

区别

在 Swift 中编译器会对 AnyObject 实例的方法调用做检查,还会返回一个 Optional 的结果。

原理

public typealias AnyObject
// The protocol to which all class types implicitly conform.

由定义就可以看出它就是一个接口,所有的 class 都隐式地实现了这个借口。所以 AnyObject 只适用于 class 类型。但是 swift 中的基本类型都是 struct 类型,并不能用 AnyObject 来表示。所以官方又提出了一个更特殊的 Any 类型,它除了 class 以外还可以表示其他类型,可以说是任意类型(包括 struct,enum,func等)。

import UIKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let swiftArr = ["a", "b", "c"]
        let swiftStr = "hello world"
        
        var array = [AnyObject]()
        array.append(swiftArr)
        array.append(swiftStr)
    }
}

这种写法是会报错的,String 不符合预期类型 AnyObject,并且系统提示了我们怎么修改:

Argument type 'String' does not conform to expected type 'AnyObject' Insert ' as AnyObject'

按提示修改后:

import UIKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let swiftArr = ["a", "b", "c"]
        let swiftStr = "hello world"
        
        var array = [AnyObject]()
        array.append(swiftArr as AnyObject)
        array.append(swiftStr as AnyObject)
    }
}

这里我们显示的将 swift 中的 String 和 Array 转成了 AnyObject。实际上 array 里面的元素已经变成了 NSString 和 NSArray 了。

当然我们还有另外的方式解决此问题,用 Any。

import UIKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
        
        let swiftArr = ["a", "b", "c"]
        let swiftStr = "hello world"
        
        var array = [Any]()
        array.append(swiftArr)
        array.append(swiftStr)
    }
}

可以看到结果全部是 swift 中的原生类型:

注意

  • 只是用 swift 类型而不转为 Cocoa 类型是会提升性能的,所以我们最好还是使用原生类型。
  • 在 OC 和 swift 混编的工程中使用 AnyObject 和 Any 是在所难免的,但我们要尽量避免使用这两者,swift 中最好明确地指出确定的类型。
  • 如果我们的代码经常用到这两者,意味着代码可能在结构和设计上存在问题。