【文末有惊喜!】利用Swift API可用性解决App Extension无法编译

本文探讨了Xcode 12.5版本后,如何解决因启用APPLICATION_EXTENSION_API_ONLY而导致的编译错误问题。介绍了使用Swift的@available和Objective-C的NS_EXTENSION_UNAVAILABLE_IOS进行API标记的方法,并详细讨论了函数调用链、协议函数、Objective-C函数API以及重写系统方法API的具体处理方式。

  

本文字数:2832

预计阅读时间:20分钟

目录

  • 问题

  • 探索

  • 平台可用性

  • API可用性

  • #available

  • 函数调用链

  • 协议函数

  • Objective-C函数API

    • 普通API

    • 重写系统方法的API

问题

从Xcode12.5开始,苹果要求所有的Extension Target必须设置APPLICATION_EXTENSION_API_ONLY为true,否则将会导致编译错误“Application extensions and any libraries they link to must be built with the APPLICATION_EXTENSION_API_ONLY build setting set to YES”;但是我们通常会在主工程和Extension之间使用Framework或其他方式共享代码,这些代码中使用了非extension-only API,所以导致问题出现,本篇文章将探讨如何解决这个问题。

探索

我们以一个具体的工程结构为例,如下图所示:

我们的主工程Host App中,创建了一个Share Extension的扩展Target做分享相关的操作;另外,为了模块化,我们有一个Library工程包含所有的基础组件和Fundation扩展方法,NetworkService工程包含网络请求相关的功能封装和处理,他们都被编译为Framework供主工程和Share Extension共同使用;

我们首先需要把Share ExtensionLibraryNetworkService这三个工程的Build Setting中APPLICATION_EXTENSION_API_ONLY设置为true;由于我们在LibraryNetworkService中都使用了UIApplication.shared.openUIApplication.shared.keyWindow这类非extension-only的API,所以编译这两个子工程是无法通过的。

首先想到的解决办法是代码拆分,我们可以把Library按是否使用extension-only API进行拆分,拆分成两个工程LibrayLibraryExtensionLibrayExtension中包含符合extension-only的API,提供给Share Extesnion使用;Library中包含其他不设限API,提供给主工程或其他非Extesnion工程使用;

然后NetworkService也采用相同方法进行改造,这种方式是可以解决问题的,但是除了拆分代码创建新工程的代价,也会带来很多额外的工作量;比如主工程中Host App,原来都是引用import Library,现在就需要逐个修改确认,是使用Libray还是LibraryExtension,或者添加同时引用两个,这对于一个已有的大工程来说,是一个不小的工作量;

另外,笔者在搜索这个问题的解决方式时,发现swift-cast网站作者提供了一个另外的解决方案:使用ACTION_EXTENSION;如果你的App恰好使用的是Action Extension可以采用这个方式去尝试:

#if !ACTION_EXTENSION
    //codes that don't obey extension-only API requests
#else
    //normal codes 
#endif 

显然,上面的两种方式都有各自的局限性,所以我们需要寻找更加广泛使用的解决方案,改动更小的解决方案;Swift语言提供了API可用性的标识,这个功能能够解决我们所遇到的问题,下面我们先来了解一下Swift的API可用性。

Swift的API可用性(API availability)

在Swift中使用@available可以标记API的可用性信息,比如是否API在某个版本被废弃,这个API需要的Swift版本大于5.4才能使用,等等。我们来看几个具体方面:

平台可用性

@available(iOS 13.0, OSX 10.15, *)
@available(tvOS, unavailable)
@available(watchOS, unavailable)
public struct SearchField: View{
    ...
}

在SwiftUI中我们定义一个SearchField组件,就需要对其适用的平台和系统版本进行限制,上面这段代码表明,SearchField适用于大于等于iOS13OSX 10.15版本,同时针对tvOSwatchOS都是不可用的。

具体注释定义如下:

@available(platform version , platform version ..., *)
  • platform:指定具体适用的平台,比如iOS, macCatalyst, macOS/OSX, tvOS 或者 watchOS还可以指定适用的扩展Extension Target,比如ApplicationExtensionmacOSApplicationExtension,这是解决文章开头提出的问题的重要工具,这部分稍后将详细介绍

  • version:具体的数字,可以由一位、两位或三位正整数通过点号(.)分割组成,分别代表主版本号,次版本号,补丁版本号;

  • 可以有多个platform+version组成,之间用逗号分隔(,),比如@available(iOS 13.0, OSX 10.15, *)

  • 星号(*),表示该API可用于所有其他平台。为了处理潜在的未来平台,平台可用性注释总是需要一个星号;

API可用性

评论 2
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值