保持NSUserActivity向后兼容Xcode 9
使用Xcode 10(beta 6),我可以毫无困难地编写和运行以下代码:
import Intents func test() { let activity = NSUserActivity(activityType: "com.activtiy.type") activity.title = "Hello World" activity.isEligibleForSearch = true activity.isEligibleForHandoff = false if #available(iOS 12.0, *) { activity.isEligibleForPrediction = true activity.suggestedInvocationPhrase = "Say something" } print(activity) }
从iOS 12开始,添加了.isEligibleForPredictions
和.suggestedInvocationPhrase
属性,因此Xcode 10可以使用if #available
条件使代码本身向后兼容。
但是,我想确保此代码向后兼容早期版本的Xcode。 在Xcode 9中运行时,我收到以下错误:
if #available(iOS 12.0, *) { // ERROR: Value of type 'NSUserActivity' has no member 'isEligibleForPrediction' activity.isEligibleForPrediction = true // ERROR: Value of type 'NSUserActivity' has no member 'suggestedInvocationPhrase' activity.suggestedInvocationPhrase = "Say something" }
这似乎是因为#available
宏实际上是在运行时解析的,因此所有包含的代码仍然需要成功编译。
有没有办法告诉编译器在构建iOS 11或使用Xcode 9时忽略这两行代码?
Xcode 10使用Swift 4.2,而Xcode 9使用Swift 4.1。 所以你可以在编译时使用这些知识:
func test() { let activity = NSUserActivity(activityType: "com.activtiy.type") activity.title = "Hello World" activity.isEligibleForSearch = true activity.isEligibleForHandoff = false #if swift(>=4.2) // compile-time check if #available(iOS 12.0, *) { // run-time check activity.isEligibleForPrediction = true activity.suggestedInvocationPhrase = "Say something" predictionApiAvailable = true } #endif print(activity) }
(这个答案假设您在Xcode 10上使用Swift 4.2。)
您正确使用的可用性条件(如果#available)在运行时进行评估,但编译器将使用此信息来提供编译时安全性(如果您调用的是API,它将会发出警告存在最小部署目标)。
对于有条件编译代码,您必须使用条件编译块
#if compilation condition statements #endif
要根据Xcode版本有条件地构建代码,可以在构建设置中包含一个Active Compilation Condition(一个-D swift编译标志),它的名称是根据Xcode版本动态创建的。 然后使用它作为编译条件。 Xcode已经提供了解析为当前Xcode版本的构建设置XCODE_VERSION_MAJOR(例如,Xcode 9的0900)。 因此,您可以添加名为XCODE_VERSION _ $(XCODE_VERSION_MAJOR)的活动编译条件,该条件将解析为Xcode 9的标志XCODE_VERSION_0900。
然后您可以使用以下方法有条件地编译代码:
#if XCODE_VERSION_0900 statements #endif
我这里有一个示例项目