问题描述
为了提高工程的编译速度,准备把第三方库编译成 Framework
。这样就可以省掉编译这些第三方库的很多时间了。
但是还涉及到一些第三方库做了一些改动,比如我们在 AFNetworking
在发送请求之前,把发送的 JSON
字符串进行自定义加密。
if (parameters) {
if (![mutableRequest valueForHTTPHeaderField:@"Content-Type"]) {
[mutableRequest setValue:@"application/json" forHTTPHeaderField:@"Content-Type"];
}
NSString *isEnc = [GBAppRunEnvironmentManager gb_shareManager].currentAppRequestEncryption;
BOOL isOpenSOA = [SOAAppRunEnvironmentManager gb_shareManager].isOpenSOA;
if (isOpenSOA) {
isEnc = [SOAAppRunEnvironmentManager gb_shareManager].currentAppRequestEncryption;
}
/**
是否需要加密
1、看是否系统有打开设置 1 (一般是正式环境下的)
2、看看是否是 我们 Gearbest 的网址 (可能后期其他额外请求不需要用到)
*/
if ([isEnc isEqualToString:@"1"] && [request.URL.absoluteString rangeOfString:@"gearbest.com"].location != NSNotFound) {
NSString *authCode = [[(NSDictionary *)parameters yy_modelToJSONString] authCodeType:NSStringAuthCodeEncoded key:kGBAuthCodeServer];
[mutableRequest setHTTPBody:[authCode dataUsingEncoding:NSUTF8StringEncoding]];
}else {
[mutableRequest setHTTPBody:[NSJSONSerialization dataWithJSONObject:parameters options:self.writingOptions error:error]];
}
//[mutableRequest setHTTPBody:[NSJSONSerialization dataWithJSONObject:parameters options:self.writingOptions error:error]];
}
这样就需要在AFNetWorking
里面用到我们自己工程的类。
如果把改过的代码进行编译,一定会抱错的,因为找不到对应的文件。
首先想到就是 __has_include
这个判断 如果工程里面有这个类,就去执行对应的方法。
为了验证这个方案是否可行,我创建一个测试的 Framework
叫做 TestFramework
。
起初我是这么写的
#import "ClassA.h"
#if __has_include("ViewController.h")
#import "ViewController.h"
#define CanLogin
#endif
@implementation ClassA
+ (void)login {
#ifdef CanLogin
ViewController *controller = [[ViewController alloc] init];
[controller loginSuccess];
#endif
}
@end
但是编译就会抱错了,错误如下。
Undefined symbols for architecture x86_64:
"_OBJC_CLASS_$_ViewController", referenced from:
objc-class-ref in ClassA.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
意思应该是连接不到这个 ViewController
的类。
经过谷歌,应该是 Framework
在编译的时候就通过__has__include
查找文件是否存在。
这个不是运行时才检测的,所以在 Framework
进行编译打包的时候已经找不到头文件。所以那样写也不正确吧。
经过群众大神的指导说是用 runtime
,这个通过字符串创建对象,用消息发送调用方法是可行的。但是对于工程写这样的代码确实是不雅观。
搜索了很久也没有其他的方法,貌似只有用 runtime
才可以让 Framework
正常的编译打包。
把代码修改成下面
+ (void)login {
Class className = NSClassFromString(@"ViewController");
if (className) {
id controller = [[className alloc] init];
objc_msgSend(controller,NSSelectorFromString(@"loginSuccess"));
}
}
需要注意的一点是工程默认不支持objc_msgSend多参数的,需要设置下面才可以。
95E32E26-F274-4D8A-AC49-97BB951666D6起先编译之后创建工程,拖入编译成功的 Framework
。一调用就报找不到文件。
群里面大神让我执行下面命令
lipo -info xxx
查看是否支持模拟器框架,但是竟然查询不出来。
B93F5CB1-67A6-4117-BC9C-5DC018A8B1CE我突然想起来了,是不是编译选择了真机。
去看了一下果然是 Xcode 默认选择了真机,所以刚才编译出来的就只能真机才能用。
我切换成模拟器,再次打包运行果然可以了。
解决办法
如果Framework
真的需要使用第三方的类,可以托管在 Cocoapods
。或者如上面使用 runtime
.
对于 Framework
编译真机和模拟器合并的可以交给 Carthage
完成,对于 之前 Cocoapods
依赖转成 Carthage
托管的还需要研究。
网友评论