> 技术文档 > iOS18+Xcode16: openURL弃用导致微信分享无响应的问题_ios18 openurl

iOS18+Xcode16: openURL弃用导致微信分享无响应的问题_ios18 openurl


背景

最近遇到个问题,当Xcode升级到16,且设备升级到iOS 18后,使用微信分享无法调起微信,控制台输出:
BUG IN CLIENT OF UIKIT: The caller of UIApplication.openURL(_:) needs to migrate to the non-deprecated UIApplication.open(_:options:completionHandler:). Force returning false (NO).

这个问题原因很简单,是因为在iOS 18以后,openURL:完全被弃用,需要用openURL:options:completionHandler:替代。

实际上在iOS10以后,Apple就已经在Xcode中提示openURL:将会被弃用,需要用openURL:options:completionHandler:替代。但实际上在iOS 17之前openURL:一直是可以用的,所以很多开发者包括很多知名的 SDK都忽略了,直到本次iOS 18完全弃用,导致问题的出现。

解决方案

解决方案也很简单:
1.如果代码是自己的写的,全文搜索并替换相关方法即可。
2.如果是第三方SDK,更新SDK即可。或者手动替换SDK里面相关方法。

特例

我遇到的问题比较奇葩,是某个第三方SDK里面出现了openURL:方法,但是这个SDK已经完全停止维护,几年不更新了,而且这个SDK不是源码模式,而是编译成了.a文件,根本没办法看到他的代码,更没办法手动替换它的代码,这特么就尴尬了。找了很久,终于找到替代方案,那就是利用runtime来替换相关方法。

Objective-C版本参考以下方案:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions{ [AppDelegate hookOldOpenUrl:AppDelegate.class];}- (BOOL)g_openURL:(NSURL*)url{ [UIApplication.sharedApplication openURL:url options:nil completionHandler:nil]; return YES;}+ (void)hookOldOpenUrl:(Class)targetCls{ Class cls = [UIApplication class]; if (cls) { Method originalMethod =class_getInstanceMethod(cls, @selector(openURL:)); Method swizzledMethod =class_getInstanceMethod(targetCls, @selector(g_openURL:)); if (!originalMethod || !swizzledMethod) { return; } IMP originalIMP = method_getImplementation(originalMethod); IMP swizzledIMP = method_getImplementation(swizzledMethod); const char *originalType = method_getTypeEncoding(originalMethod); const char *swizzledType = method_getTypeEncoding(swizzledMethod); class_replaceMethod(cls,@selector(openURL:),swizzledIMP,swizzledType); class_replaceMethod(cls,@selector(g_openURL:),originalIMP,originalType); }}

Swift版本参考以下方案:

func hookOldOpenUrl(tragetCls:AnyClass){ let cls = UIApplication.self let originalSelector = #selector(UIApplication.openURL(_:))  let swizzledSelector = #selector(g_openURL) let originalMethod = class_getInstanceMethod(cls, originalSelector) let swizzledMethod = class_getInstanceMethod(tragetCls, swizzledSelector) let didAddMethod: Bool = class_addMethod(cls, originalSelector, method_getImplementation(swizzledMethod!), method_getTypeEncoding(swizzledMethod!)) if didAddMethod {  class_replaceMethod(cls, swizzledSelector, method_getImplementation(originalMethod!), method_getTypeEncoding(originalMethod!)) } else {  method_exchangeImplementations(originalMethod!, swizzledMethod!) }}@objc func g_openURL(url:URL)->Bool{ UIApplication.shared.open(url) return true}

再次运行即可修复该问题。