Any,AnyObject
Any 和AnyObject的区别就是Any指所有类型(泛型),AnyObject指类(可以叫类泛型)
as? as! as
as 是强制转换,就是强转符,但因为强转是会失败的,所以as? as! 代表可失败的强制转换符
class Person {
func eat() {
print("eat")
}
}
struct Human {
func run() {
print("run")
}
}
var h = Human()
var p = (h as? Person)
p?.eat()
比如这个,把h强制转换为Person就不会成功,所以p是nil,所以调用eat会失败
class Person {
func eat() {
print("eat")
}
}
struct Human {
func run() {
print("run")
}
}
var h = Human()
var p = (h as! Human)
print(p)
p.run()
as!和as的用法一样,是直接调用,不过as在明显是失败的情况下,通不过编译器,as!是可以的
X.self X.type AnyClass is
在说这些之前先介绍下类的基本结构类本身是一个8个字节的指针,这个指针指向的是一个堆内容,第一位也是个8字节的指针,第二位是引用计数,后面是存储属性 第一位8个字节指针指向的是元类信息,比如父类继承下来的方法地址,类是什么属性等等
而X.self 就是表示的元类的指针,并且X.self就是X.type类型 ,比如 var type = Person.self 这个type的类型就是Person.type
class Person {
var age = 10
}
var p = Person()
var type:AnyClass = Person.self
print(MemoryLayout.size(ofValue: type)) // 8
print(Mems.ptr(ofVal: &p)) //0x000000010000c380
print(Mems.ptr(ofVal: &type)) // 0x000000010000c388
print("")
可以看出是8个字节的指针,然后打断点查看内存
可以看出p的内存地址是0x000000010000c380 type的内存地址是0x000000010000c388,这就是两个挨着的内存,然后打印出来发现都是一个指针,p指向的是存放类信息的地址,然后查看这个堆地址,发现第一个地址就是type的指针指向的地址,所以破案了,X.self 就是指向元类信息的指针
另外AnyClass就是AnyObject.Type的别名,而AnyObject.Type就是元类指针的泛型
所以总结下,X.type表示的是某个类的元类指针类型,是个类型,X.self表示的是某个类的元类指针,AnyClass是可以代表所有元类类型指针的泛型