Dodanie dodatkowych informacji do już świetnych odpowiedzi.
stworzyłem trzy programy i spojrzał na wygenerowanym montażu od siebie:
obj1.swift
import Foundation
class MyClass { }
let obj = MyClass()
obj2.swift
import Foundation
class MyClass { }
let obj: AnyObject = MyClass()
obj3.swift
import Foundation
class MyClass { }
let obj: AnyObject = MyClass()
obj.isKindOfClass(MyClass.self)
różnice między obj1 i obj2 są trywialne. Wszelkie instrukcje, które obejmują typ obiektu mają różne wartości:
movq %rax, __Tv3obj3objCS_7MyClass(%rip)
# ...
globl __Tv3obj3objCS_7MyClass .globl __Tv3obj3objPs9AnyObject_
.zerofill __DATA,__common,__Tv3obj3objCS_7MyClass,8,3
# ...
.no_dead_strip __Tv3obj3objCS_7MyClass
vs
movq %rax, __Tv3obj3objPs9AnyObject_(%rip)
# ...
.globl __Tv3obj3objPs9AnyObject_
.zerofill __DATA,__common,__Tv3obj3objPs9AnyObject_,8,3
# ...
.no_dead_strip __Tv3obj3objPs9AnyObject_
Pełna edycja here.
To było interesujące dla mnie. Jeśli jedynymi różnicami między tymi dwoma plikami są nazwy typu obiektu, dlaczego obiekt zadeklarowany jako AnyObject
może wykonywać selektor Objective-C?
obj3 pokazuje jak selektor isKindOfClass:
wypala:
LBB0_2:
# ...
movq __Tv3obj3objPs9AnyObject_(%rip), %rax
movq %rax, -32(%rbp)
callq _swift_getObjectType
movq %rax, -8(%rbp)
movq -32(%rbp), %rdi
callq _swift_unknownRetain
movq -24(%rbp), %rax
cmpq $14, (%rax)
movq %rax, -40(%rbp)
jne LBB0_4
movq -24(%rbp), %rax
movq 8(%rax), %rcx
movq %rcx, -40(%rbp)
LBB0_4:
movq -40(%rbp), %rax
movq "L_selector(isKindOfClass:)"(%rip), %rsi
movq -32(%rbp), %rcx
movq %rcx, %rdi
movq %rax, %rdx
callq _objc_msgSend
movzbl %al, %edi
callq __TF10ObjectiveC22_convertObjCBoolToBoolFVS_8ObjCBoolSb
movq -32(%rbp), %rdi
movb %al, -41(%rbp)
callq _swift_unknownRelease
xorl %eax, %eax
addq $48, %rsp
# ...
LBB6_3:
.section __TEXT,__objc_methname,cstring_literals
"L_selector_data(isKindOfClass:)":
.asciz "isKindOfClass:"
.section __DATA,__objc_selrefs,literal_pointers,no_dead_strip
.align 3
"L_selector(isKindOfClass:)":
.quad "L_selector_data(isKindOfClass:)"
Diff między obj2 i obj3 here.
isKindOfClass
jest wysyłany jako metoda dynamicznie wysyłana, tak jak w przypadku _objc_msgSend
. Oba obiekty są narażone na Objective-C jako SwiftObject
(.quad _OBJC_METACLASS_$_SwiftObject
), deklarując typ obiektu jako AnyObject
uzupełniający most do NSObjectProtocol
.
Ale "AnyObject" jest zdefiniowany jako '@kjc public protocol AnyObject {}' Czy to oznacza, że wszystkie protokoły '@ objc' są zgodne z' NSObjectProtocol'? A co z "Any"? – JAL
To wcale tak nie znaczy. Mówię to, co mówię, a nie jakaś inna rzecz. – matt
Mówisz, że "klasa pozbawiona podstaw, taka jak MyClass, jest niemożliwa". Więc mówisz, że 'MyClass' dziedziczy po' SwiftObject'? Dlaczego tylko rzutowanie na "AnyObject" ujawnia zgodność z 'NSObjectProtocol'? – JAL