[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,这个时候会把组件间的调用去查表之类的进行连接,如果有找不到的接口调用会报错哒。

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

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