[iOS] 二进制打包引发的crash思考

这周遇到一个很神奇的crash,报错是找不到 responser,然后惊讶的发现了一些二进制打包的坑,也是太久没写小水文儿了于是来玩儿一下~

  • 背景是酱紫的:
    我们有个 protocol A,声明了很多属性,然后有一个 proxy 实现了这个 protocol A,然鹅它其实木有实现协议里面的属性,而是重写了 forwarding 方法,当你读取他的某个属性的时候,他会调用自己的一个方法,获取到自己持有的另一个对象来返还给 forwardInvocation

  • 如何找到要返回哪个对象的呢?
    是通过 runtime 找到 property_list和属性对应的 class,然后存到一个map里面,当外面找某个属性的时候,它会通过map里面找到这个属性对应的 class,再从自己持有的一个字典里面找到这个 class 的对象。

然后这个crash是在一个会给这个 protocol A 新增一个属性的分支上面发生的,报的crash是调用了这个实现了 protocol A 的对象forwarding的时候没能找到响应这个新属性的对象,然鹅肯定的是这个字典里一定有这个新属性class对应的对象。

  • 问题出在哪里呢?
    这个问题当时我木有反应过来是为什么其实,后来在大佬的帮助下了解到了为啥。

这个 protocolA 是在其他库里面也会用到的,protocol 在编译的时候会被搞成 struct,于是每个用到这个 protocol 的binary库都有一份自己的 struct,于是在我们 runtime 找这个 protocol 的属性的时候,可能用到了其他二进制包里面的 struct,但是这个 struct 是旧的,没有加新的属性的,那么在生成属性和class对应的map的时候就木有找到这个属性,所以才会crash。

  • 解决方式:
  1. 把所有用到了这个 protocol 的库都加入到开发库新发一个版,之前的缓存就不生效啦
  2. 由于被其他库引用的 protocol 会被其他库的二进制缓存,那么我们可以在内部新建一个protocol B 继承自 protocol A,并且把属性重写一遍(因为不是很清楚继承的原理),这样内部用的 protocol B 一定是可以runtime找到这个属性的啦~ 但是要保证其他库不能 import 到这个protocol B 哦~
#define propertiesDeclaration \
@property (nonatomic, readonly) XXXClass *xxx;\
@property (nonatomic, readonly) YYYClass *yyy;

------

@protocol ProtocolA <NSObject>

propertiesDeclaration

@end

------

@protocol ProtocolB <ProtocolA>

propertiesDeclaration

@end

※ Protocol编译后是个什么样子的struct呢?

我们用clang来康康包含protocol的类是咋编译的:

xcrun -sdk iphonesimulator clang -rewrite-objc 文件名.m

文件里面定义了这么一个协议:

@protocol RStoreObserver <NSObject>

-(void)onStateChanged:(RState *)newState;

@end

产物是酱紫的,里面的RStoreObserver木有啥感觉,然后就是所有都是struct毕竟其实OC底层都是靠C搭起来的:

struct _protocol_t;

struct _objc_method {
    struct objc_selector * _cmd;
    const char *method_type;
    void  *_imp;
};

struct _protocol_t {
    void * isa;  // NULL
    const char *protocol_name;
    const struct _protocol_list_t * protocol_list; // super protocols
    const struct method_list_t *instance_methods;
    const struct method_list_t *class_methods;
    const struct method_list_t *optionalInstanceMethods;
    const struct method_list_t *optionalClassMethods;
    const struct _prop_list_t * properties;
    const unsigned int size;  // sizeof(struct _protocol_t)
    const unsigned int flags;  // = 0
    const char ** extendedMethodTypes;
};

※ 其他的一些小插曲

我们远端会在做一些check防止我们改了接口,其他二进制编译不过。于是我就遇到了一个check失败的问题,虽然是他们check脚本的bug。

这个背景比较简单,就是如果你在development pod里面定义了一个宏,然后其他非开发仓引用了并且出了二进制包。这个时候我把我们定义的宏改成 inline 函数,但其实里面调用的都是一样的。这个时候虽然本地木有报错,但是远端的出包都会挂掉。或者你把定义这个别的库用到的宏的.h文件里面删掉几个宏之类的可能也会引发报错。

于是我和胖友们讨论了一下:(以下都是个人理解不保证正确哦)

  • 宏在预编译会被展开,生成的二进制不应该因为宏而有变化,除非是宏展开以后的接口变化。那如果我import的不是自己库里面的宏,生成二进制的时候还是展开的咩?
    应该是的,宏应该不会像函数调用那种需要rebase link之类的。
  • 复习一下,二进制之间的函数调用,是每个库都会有一个自己对外的接口表,我们build最后一步会做link,这个时候会把组件间的调用去查表之类的进行连接,如果有找不到的接口调用会报错哒。

我决定要刷一遍程序员自身修养了。。。

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 今天感恩节哎,感谢一直在我身边的亲朋好友。感恩相遇!感恩不离不弃。 中午开了第一次的党会,身份的转变要...
    迷月闪星情阅读 10,617评论 0 11
  • 彩排完,天已黑
    刘凯书法阅读 4,281评论 1 3
  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 126,147评论 2 7