本系列博客是本人的源码阅读笔记,如果有 iOS 开发者在看 runtime 的,欢迎大家多多交流。为了方便讨论,本人新建了一个微信群(iOS技术讨论群),想要加入的,请添加本人微信:zhujinhui207407,【加我前请备注:ios 】,本人博客http://www.kyson.cn 也在不停的更新中,欢迎一起讨论
本文完整版详见笔者小专栏:https://xiaozhuanlan.com/runtime
前言
今天是最后一篇讲解 _read_images 的文章了,相信大家已经厌烦一直在讲 _read_images 这个方法了。但毕竟这个方法确实很重要,所以笔者才在这里着墨最多。今天笔者将最后的几个代码块贴出来,大家一起看看吧。
详细
获取所有的类引用
if (!noClassesRemapped()) {
for (EACH_HEADER) {
Class *classrefs = _getObjc2ClassRefs(hi, &count);
for (i = 0; i < count; i++) {
//这里笔者打印出了相关信息
printf("ClassRefs:%s",classrefs[i]->mangledName());
fflush(stdout);
remapClassRef(&classrefs[i]);
}
// fixme why doesn't test future1 catch the absence of this?
classrefs = _getObjc2SuperRefs(hi, &count);
for (i = 0; i < count; i++) {
remapClassRef(&classrefs[i]);
}
}
}
获取所有的方法引用
static size_t UnfixedSelectors;
for (EACH_HEADER) {
if (hi->isPreoptimized()) continue;
// printf("name:%s\n",hi->fname());
// fflush(stdout);
bool isBundle = hi->isBundle();
SEL *sels = _getObjc2SelectorRefs(hi, &count);
UnfixedSelectors += count;
for (i = 0; i < count; i++) {
const char *name = sel_cname(sels[i]);
// printf("name:%s\n",name);
// fflush(stdout);
sels[i] = sel_registerNameNoLock(name, isBundle);
}
}
获取所有的协议列表
for (EACH_HEADER) {
//以下两行是笔者加的调试信息
printf("class name:%s\n",hi->fname());
fflush(stdout);
extern objc_class OBJC_CLASS_$_Protocol;
Class cls = (Class)&OBJC_CLASS_$_Protocol;
assert(cls);
NXMapTable *protocol_map = protocols();
bool isPreoptimized = hi->isPreoptimized();
bool isBundle = hi->isBundle();
protocol_t **protolist = _getObjc2ProtocolList(hi, &count);
for (i = 0; i < count; i++) {
//以下两行是笔者加的调试信息
printf("protocol :%s\n",protolist[i]->nameForLogging());
fflush(stdout);
readProtocol(protolist[i], cls, protocol_map,
isPreoptimized, isBundle);
}
}
这个代码段笔者要稍微说一下,因为这个变量名可能大家有点不能理解:
OBJC_CLASS_$_Protocol
全局搜索,我们发现,其实是定义在 汇编文件
objc-sel-table.s
中的。笔者将打印出来的信息放到笔者的 GitHub 中了。供大家查阅,地址在这里:
protocollist.txt
大家可能留意到文件最后有个 KysonHere 协议,其实是笔者自己加的,这里笔者将自己写的测试类,以及协议代码贴出来:
@protocol KysonHere
@optional
-(void) kysonIsHer;
@end
也算是笔者的一段测试代码了。
获取所有的协议引用列表
for (EACH_HEADER) {
protocol_t **protolist = _getObjc2ProtocolRefs(hi, &count);
for (i = 0; i < count; i++) {
//以下两行为笔者的测试代码
printf("ref protocol :%s\n",protolist[i]->nameForLogging());
fflush(stdout);
remapProtocolRef(&protolist[i]);
}
}
返回的是空。
获取所有的分类列表
// Discover categories.
for (EACH_HEADER) {
category_t **catlist = _getObjc2CategoryList(hi, &count);
bool hasClassProperties = hi->info()->hasCategoryClassProperties();
for (i = 0; i < count; i++) {
category_t *cat = catlist[i];
Class cls = remapClass(cat->cls);
if (!cls) {
catlist[i] = nil;
continue;
}
bool classExists = NO;
if (cat->instanceMethods || cat->protocols
|| cat->instanceProperties)
{
addUnattachedCategoryForClass(cat, cls, hi);
if (cls->isRealized()) {
remethodizeClass(cls);
classExists = YES;
}
}
if (cat->classMethods || cat->protocols
|| (hasClassProperties && cat->_classProperties))
{
addUnattachedCategoryForClass(cat, cls->ISA(), hi);
if (cls->ISA()->isRealized()) {
remethodizeClass(cls->ISA());
}
}
}
}
总结
_read_images 方法终于告一段落了,希望大家有所收获!