I have a protocol defined:
protocol Usable {
func use()
}
and a class that conforms to that protocol
class Thing: Usable {
func use () {
println ("you use the thing")
}
}
I would like to programmatically test whether or not the Thing class conforms to the Usable protocol.
let thing = Thing()
// Check whether or not a class is useable
if let usableThing = thing as Usable { // error here
usableThing.use()
}
else {
println("can't use that")
}
But I get the error
Bound value in a conditional binding must be of Optional Type
If I try
let thing:Thing? = Thing()
I get the error
Cannot downcast from 'Thing?' to non-@objc protocol type 'Usable'
I then add @objc
to the protocol and get the error
Forced downcast in conditional binding produces non-optional type 'Usable'
At which point I add ?
after the as
, which finally fixes the error.
How can I achieve this functionality with conditional binding with a non-@objc protocol, the same as in the "Advanced Swift" 2014 WWDC Video?
You can get it to compile by making the cast as Usable? instead of as Usable, like this:
As metioned in the Swift doc, the
is
operator is the guy you need for the job:Therefore, the following test would normally be what you need:
However, as the doc specifies, Swift can detect at compile time that the expression is always true and declares an error to help the developer.
You are getting
because
thing as Usable
must return an optional type so making itas?
should solved the problem. Unfortunately, the error still persisted for some odd reason. Anyway, a workaround I found to get it to work is to extract out the variable assignment inside the if statementswift protocols does not work in Playgrounds in the first beta, try to build a real project instead.
This works for me in the playground