I need to keep a collection of Swift metatypes and write a function which will check if a given object is an instance of one of them. I can do that easily in Java:
Class c = x.getClass();
c.isInstance(someObj)
However, I have no idea how to do that in Swift:
var isInt = 7 is Int.Type // compiles
let x = Int.self
var isInt = 7 is x // compiler error - Use of undeclared type 'x'
Is this even possible to be done in Swift?
Checking TypeUse the type check operator ( is ) to check whether an instance is of a certain subclass type. The type check operator returns true if the instance is of that subclass type and false if it's not.
A metatype type refers to the type of any type, including class types, structure types, enumeration types, and protocol types. The metatype of a class, structure, or enumeration type is the name of that type followed by .
metatype (plural metatypes) (biology) A topotype of a species which is confirmed as belonging to that species by the author who originally described it. (programming) The type of a type.
Unfortunately, you can currently only use a named type with the is
operator, you cannot yet use an arbitrary metatype value with it (although really IMO you ought to be able to).
Assuming you have control over the creation of the metatypes that you want to compare against, one solution that achieves the same result would be to create a wrapper type with an initialiser that stores a closure that performs the is
check against a generic placeholder:
struct AnyType {
let base: Any.Type
private let _canCast: (Any) -> Bool
/// Creates a new AnyType wrapper from a given metatype.
/// The passed metatype's value **must** match its static value,
/// i.e `T.self == base`.
init<T>(_ base: T.Type) {
precondition(T.self == base, """
The static value \(T.self) and dynamic value \(base) of the passed \
metatype do not match
""")
self.base = T.self
self._canCast = { $0 is T }
}
func canCast<T>(_ x: T) -> Bool {
return _canCast(x)
}
}
protocol P {}
class C : P {}
class D : C {}
let types = [
AnyType(P.self), AnyType(C.self), AnyType(D.self), AnyType(String.self)
]
for type in types {
print("C instance can be typed as \(type.base): \(type.canCast(C()))")
print("D instance can be typed as \(type.base): \(type.canCast(D()))")
}
// C instance can be typed as P: true
// D instance can be typed as P: true
// C instance can be typed as C: true
// D instance can be typed as C: true
// C instance can be typed as D: false
// D instance can be typed as D: true
// C instance can be typed as String: false
// D instance can be typed as String: false
The only limitation of this approach is that given we're performing the is
check with T.self
, we have to enforce that T.self == base
. For example, we cannot accept AnyType(D.self as C.Type)
, as then T.self
would be C.self
while base
would be D.self
.
However this shouldn't be a problem in your case, as we're just constructing AnyType
from metatypes that are known at compile time.
If however you don't have control over the creation of the metatypes (i.e you get handed them from an API), then you're quite a bit more limited with what you can do with them.
As @adev says, you can use type(of:)
in order to get the dynamic metatype of a given instance, and the ==
operator to determine if two metatypes are equivalent. However, one problem with this approach is that it disregards both class hierarchies and protocols, as a subtype metatypes will not compare equal with a supertype metatypes.
One solution in the case of classes is to use Mirror
, as also shown in this Q&A:
/// Returns `true` iff the given value can be typed as the given
/// **concrete** metatype value, `false` otherwise.
func canCast(_ x: Any, toConcreteType destType: Any.Type) -> Bool {
return sequence(
first: Mirror(reflecting: x), next: { $0.superclassMirror }
)
.contains { $0.subjectType == destType }
}
class C {}
class D : C {}
print(canCast(D(), toConcreteType: C.self)) // true
print(canCast(C(), toConcreteType: C.self)) // true
print(canCast(C(), toConcreteType: D.self)) // false
print(canCast(7, toConcreteType: Int.self)) // true
print(canCast(7, toConcreteType: String.self)) // false
We're using sequence(first:next:)
to create a sequence of metatypes from the dynamic type of x
through any superclass metatypes it might have.
However this method still won't work with protocols. Hopefully a future version of the language will provide much richer reflection APIs that allow you to compare the relationship between two metatype values.
However, given the above knowledge of being able to use Mirror
, we can use it to lift the aforementioned restriction of T.self == base
from our AnyType
wrapper on by handling class metatypes separately:
struct AnyType {
let base: Any.Type
private let _canCast: (Any) -> Bool
/// Creates a new AnyType wrapper from a given metatype.
init<T>(_ base: T.Type) {
self.base = base
// handle class metatypes separately in order to allow T.self != base.
if base is AnyClass {
self._canCast = { x in
sequence(
first: Mirror(reflecting: x), next: { $0.superclassMirror }
)
.contains { $0.subjectType == base }
}
} else {
// sanity check – this should never be triggered,
// as we handle the case where base is a class metatype.
precondition(T.self == base, """
The static value \(T.self) and dynamic value \(base) of the passed \
metatype do not match
""")
self._canCast = { $0 is T }
}
}
func canCast<T>(_ x: T) -> Bool {
return _canCast(x)
}
}
print(AnyType(D.self as C.Type).canCast(D())) // true
The case where T.self
is a class metatype should be the only case where T.self != base
, as with protocols, when T
is some protocol P
, T.Type
is P.Protocol
, which is the type of the protocol itself. And currently, this type can only hold the value P.self
.
Ideally the following should work in your case with some changes as type(of: 7)
instead of 7
and ==
operator instead of is
. But swift has a bug which throws following error.
let x = Int.self
var isInt = type(of: 7) == x //binary operator '==' cannot be applied to two 'Int.Type' operands
Instead you can use the following code and it will work fine.
let x = Int.self
var typeOfX = type(of: 7)
var isInt = typeOfX == x //true
Apple engineers have confirmed this bug here:
Joe Groff - Twitter
First line in your question should be var isInt = 7 is Int
. Notice Int
instead of Int.Type
. Otherwise Xcode will throw below warning.
Usually in most cases you could just do,
if z is String {
//do something
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With