Block技术在iOS开发中很是流行也很方便,可是稍微疏忽就可能会产生引用没法被释放的问题,从而形成内存泄漏。那如何知道哪一个Block持有了对象并形成内存泄漏呢?bash
一个解决的方法是在程序运行时经过Xcode的Debug Memory Graph 来查看当前进程中全部生命周期内的对象。这样能够在调试时经过这个功能发现一些原本应该被释放可是却没有被释放的对象。从而肯定哪些对象有内存泄漏的嫌疑。 函数
当点击某个对象时,右边能够看出这个对象的内存分配状况以及被引用的状况,从而能够进一步跟踪确认出对象是被谁持有和引用而没有被正常的释放。spa
在上图中黑色的线部分就是对象被强引用的序列图。调试
回到主题,你能够上面的图形中看出对象ViewController2是被一个**__NSMallocBlock__** 所持有了,可是你只能看到这个Block对象的内存地址(右上角)而已。要想看这个Block所对应的实现代码时你只须要在lldb控制台输入以下信息:code
(lldb) dis -s *(void**)(0x600002f51110+16)
MyLoadTest`__27-[ViewController2 loadView]_block_invoke:
0x10c79c080 <+0>: pushq %rbp
0x10c79c081 <+1>: movq %rsp, %rbp
0x10c79c084 <+4>: subq $0x40, %rsp
0x10c79c088 <+8>: movq %rdi, -0x8(%rbp)
0x10c79c08c <+12>: movq %rdi, %rax
0x10c79c08f <+15>: movq $0x0, -0x10(%rbp)
0x10c79c097 <+23>: leaq -0x10(%rbp), %rcx
0x10c79c09b <+27>: movq %rdi, -0x20(%rbp)
复制代码
上述指令中 dis -s 地址 的做用是用来反汇编某个地址所对应符号信息以及开始一部分的汇编实现。cdn
命令中然后面的0x600002f51110 则是Block对象的地址,这里加16的意思是由于Block对象的内部偏移16个字节的位置就是Block对象所保存的执行代码的函数地址。 因此经过这个指令就能够轻松的知道是哪一个Block对象强持有了对象而不会被释放了。对象
从上面的第一张图中的源代码能够看出Block内部持有了self对象致使了对象没法被正常释放。blog
经过上述的命令能够在调试时用在任何地方来查看某个Block的函数信息。生命周期
这里须要注意的是当你在一个方法内定义了多个Block时。这些Block的函数符号的规则是:进程
-[block定义所在的方法名]_block_invoke.序号
复制代码
在方法中定义的第一个block是没有序号,然后续的则根据定义的数量从2递增。
好比下面类中的定义的四个block:
@interface CA
-(void)foo1{
void(^b)(void) =^{};
void(^b)(void) =^{};
}
-(void)foo2{
void(^b)(void) =^{};
void(^b)(void) =^{};
}
@end
复制代码
所对应的block的符号是:
-[CA foo1]_block_invoke
-[CA foo1]_block_invoke.2
-[CA foo2]_block_invoke
-[CA foo2]_block_invoke.2
复制代码