打印block中引用的所有Object对象

这篇文章是参考了欧阳大哥的实现并在此基础上进行了一些扩展,原文链接在此一种查看Block中引用的所有外部对象的实现方法。想要详细了解如何解析block中引用的对象直接查看原文链接即可。

扩展说明:在原作者的实现中,没有实现打印__block类型的引用对象,我在此进行了补充。下图是打印出block中所有的引用对象


打印block引用的外部对象

核心代码如下:

static NSString *block_description(id block, SEL _cmd) {
    struct Block_layout *blockLayout = (__bridge struct Block_layout *)block;
    struct Block_descriptor_1 *desc1 = blockLayout->descriptor;
    struct Block_descriptor_2 *desc2 = NULL;
    struct Block_descriptor_3 *desc3 = NULL;
    
    NSMutableString *printStr = [NSMutableString new];
    
    if (blockLayout->flags & BLOCK_HAS_COPY_DISPOSE) {
        desc2 = (struct Block_descriptor_2 *)(desc1 + 1);
    } else {
        return printStr;
    }
    
    if (blockLayout->flags & BLOCK_HAS_EXTENDED_LAYOUT) {
        desc3 = (struct Block_descriptor_3 *)(desc2 + 1);
    } else {
        return printStr;
    }
    
    if (desc3->layout == 0) {
        return printStr;
    }
    
    
    const char *extendLayout = desc3->layout;
    
    //如果layout小于0x1000,则将layout本身当成一个12bit的数据,此时是compact encoding
    //压缩编码方式0xXYZ: X表示强引用数量,Y表示__block引用数量,Z表示弱引用数量,对该编码解码,从而统一下面的处理逻辑
    if (extendLayout < (const char *)0x1000) {
        char compactEncoding[4] = {0,0,0,0};
        unsigned short xyz = (unsigned short)extendLayout;
        unsigned char x = (xyz >> 8) & 0xf;
        unsigned char y = (xyz >> 4) & 0xf;
        unsigned char z = (xyz & 0xf);
        
        int idx = 0;
        if (x) {
            x--;
            compactEncoding[idx++] = (BLOCK_LAYOUT_STRONG << 4) | x;
        }
        if (y) {
            y--;
            compactEncoding[idx++] = (BLOCK_LAYOUT_BYREF << 4) | y;
        }
        if (z) {
            z--;
            compactEncoding[idx++] = (BLOCK_LAYOUT_WEAK << 4) | z;
        }
        extendLayout = compactEncoding;
    }
    
    //上面的代码解码了compact encoding情况
    //下面进行layout的解析,block内部会优先把对象类型排在前面
    
    
    //当layout大于0x1000,layout就是指向了一个字符串的指针
    int index = 0;
    int objOffest = sizeof(struct Block_layout);//block内部对象的偏移
    
    char *typeArr[4] = {"strong  ","byref   ","weak    ","unretain"};
    char *byrefTypeArr[3] = {"__strong __block","__weak __block","__unsafe_unretained __block"};
    
    
    while (extendLayout[index] != '\0') {
        
        unsigned char PN = extendLayout[index];
        unsigned char P = (PN >> 4) & 0xf; // 类型描述,strong,byref,weak等等
        unsigned char N = (PN & 0xf) + 1; //对应类型的个数
        
        //只针对对象类型进行判断
        if (P >= BLOCK_LAYOUT_STRONG && P <= BLOCK_LAYOUT_UNRETAINED) {
            
            int typeIndex = P - BLOCK_LAYOUT_STRONG;
            
            if (P != BLOCK_LAYOUT_BYREF) {
                [printStr appendFormat:@"\n引用类型:%s, 引用对象:",typeArr[typeIndex]];
                for (int i = 0; i < N; i++) {
                    
                    void *objPointer = *(void **)((void *)blockLayout + objOffest);
                    id obj = (__bridge id)(objPointer);
                    
                    [printStr appendFormat:@"%@",obj];
                    if (i != N - 1) {
                        [printStr appendString:@","];
                    }
                    
                    objOffest += sizeof(void *);//因为都是对象类型,每次偏移一个指针的大小
                }
                
            } else {
                //额外处理__block封装的类型,严格来说byref不能算是对象类型,虽然它有isa,但是该值被设为了0
                [printStr appendFormat:@"\n引用类型:%s, 引用对象:",typeArr[typeIndex]];
                
                NSInteger byrefObjCount = 0;//计数byref封装obj的数量
                
                
                for (int i = 0; i < N; i++) {
                    
                    struct Block_byref *byref1 = *(struct Block_byref **)((void *)blockLayout + objOffest);
                    byref1 = byref1->forwarding;
                    if ((byref1->flags & BLOCK_BYREF_LAYOUT_MASK) != BLOCK_BYREF_LAYOUT_NON_OBJECT) {//说明是obj对象
                        int byrefTypeIndex = (((byref1->flags - BLOCK_BYREF_LAYOUT_STRONG) >> 28) & 0xf);
                        
                        void *objPointer = *(void **)((void *)byref1 + byref1->size - sizeof(void *));
                        id obj = (__bridge id)(objPointer);
                        
                        if (byrefObjCount > 0) {
                            [printStr appendString:@","];
                        }
                        [printStr appendFormat:@"%@(%s)",obj, byrefTypeArr[byrefTypeIndex]];
                        
                        byrefObjCount++;
                    }
                    
                    objOffest += sizeof(void *);//因为都是对象类型,每次偏移一个指针的大小
                }
                
                if (byrefObjCount == 0) {
                    [printStr appendString:@"无__block修饰的对象"];
                }
                
            }
            
        }
        
        index++;
    }
    
    return printStr;
}

完整代码链接BlockExtendLayout

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

推荐阅读更多精彩内容