iOS中@strongify和@weakify

983 阅读3分钟

主要是在block中使用

一、核心用法介绍

因为block一般都在对象内部声明. 如果在block内部使用了当前对象的属性,就会造成循环引用 (block拥有当前对象的地址,而当前对象拥有block的地址), 而引起内存泄露,block和当前对象都无法释放.

@weakify 将当前对象声明为weak.. 
这样block内部引用当前对象,就不会造成引用计数+1可以破解循环引用

@strongify 相当于声明一个局部的strong对象,等于当前对象.
可以保证block调用的时候,内部的对象不会释放

可以认为就是下面这样用

__weak typeof(self)weakSelf = self;
block = ^(){
    __strong typeof(weakSelf)strongSelf = weakSelf;
};

二、两点思考

细心一点,这边会有两个问题 第一,我们常用的是,只在block外面使用__weak,内部没有__strong这个对象,会有问题吗?

可能会有,
直接用__weak的话,如果block内多次使用weakSelf的话,除第一个weakSelf外,后面的值有可能被释放,
即weakSelf=nil。
而__strong能保证在block执行完毕后才释放,
保证了block内部strongSelf有值。

第二个问题,如果在block外部进行__weak 是为了在block内部使用weakSelf不强引用, 但是为什么在内部会转回__strong 那会引起循环引用吗?

不会循环引用,这里有两种理解,
1.block 内部创建的强引用,block 是不会持有它的,block 只持有外部的强应用
2. 一旦执行 block, 这个引用就必须是强引用.  防止对象被其他线程销毁掉(防止给一个不存在的对象发消息)

三、自定义宏

@strongify和@weakify这两种宏在YYKit和RAC中都有,这里不展开讲.

@strongify和@weakify这两种宏如果我们自定义的话,可以这么写

#ifndef    weakify
#if __has_feature(objc_arc)

#define weakify( x ) \\
_Pragma("clang diagnostic push") \\
_Pragma("clang diagnostic ignored \\"-Wshadow\\"") \\
autoreleasepool{} __weak __typeof__(x) __weak_##x##__ = x; \\
_Pragma("clang diagnostic pop")

#else

#define weakify( x ) \\
_Pragma("clang diagnostic push") \\
_Pragma("clang diagnostic ignored \\"-Wshadow\\"") \\
autoreleasepool{} __block __typeof__(x) __block_##x##__ = x; \\
_Pragma("clang diagnostic pop")

#endif
#endif

#ifndef    strongify
#if __has_feature(objc_arc)

#define strongify( x ) \\
_Pragma("clang diagnostic push") \\
_Pragma("clang diagnostic ignored \\"-Wshadow\\"") \\
try{} @finally{} __typeof__(x) x = __weak_##x##__; \\
_Pragma("clang diagnostic pop")

#else

#define strongify( x ) \\
_Pragma("clang diagnostic push") \\
_Pragma("clang diagnostic ignored \\"-Wshadow\\"") \\
try{} @finally{} __typeof__(x) x = __block_##x##__; \\
_Pragma("clang diagnostic pop")

#endif
#endif

四、指令说明

_Pragma操作符是干嘛的? 其实他和#pragma 指令相同的功能, 相比预处理指令#pragma,_Pragma操作符可用于宏定义中, ,#pragma不行,因为编译器会将指令中的数字符号(“#”)解释为字符串化运算符 (#)

_Pragma("clang diagnostic push") 
_Pragma("clang diagnostic ignored "-Wshadow"") 
      // your code
_Pragma("clang diagnostic pop")

这段代码的功能是: 屏蔽掉指针变更的警告: 当声明的变量遮蔽(shadow)了参数、全局变量或者是其他局部变量时,该警告选项会给我们以警告信息. 原文: declaration shadows a %select{" "local variable|" "variable in %2|" "static data member of %2|" "field of %2}1 不怎么会翻译- .-

补充

__typeof__ 一般情况下用typeof就可以了,但是如果要于ISO C兼容的话,最好是用双下划线的形式:__typeof__

五、参考地址

@weakify, @strongify 介绍 www.jianshu.com/p/3d6c4416d…

iOS开发#pragma预处理指令与_Pragma操作符 blog.csdn.net/jancywen/ar…

-Wshadow及相关介绍 fuckingclangwarnings.com

作者:Lin__Chuan 链接:www.jianshu.com/p/bc794fa07… 来源:简书