block作为类的属性时用copy

1. block作为类的属性时用copy

Block属性的声明,首先需要用copy修饰符,因为只有copy后的Block才会在堆中,栈中的Block的生命周期是和栈绑定的

<栈 :由系统维护的局部变量 是存在栈上的,其生命周期随函数的生命周期>

<堆 :由程序员申请空间地址,由程序员手动释放,生命周期受到程序员控制>

使用retain也可以,因为block的retain行为默认是用copy的行为实现的,block变量默认是声明为栈变量的,为了能够在block的声明域外使用,所以要把block拷贝(copy)到堆,所以说为了block属性声明和实际的操作一致,最好声明为copy。

另一个需要注意的问题是关于线程安全,在声明Block属性时需要确认“在调用Block时另一个线程有没有可能去修改Block?”这个问题,如果确定不会有这种情况发生的话,那么Block属性声明可以用nonatomic。如果不肯定的话(通常情况是这样的),那么你首先需要声明Block属性为atomic,也就是先保证变量的原子性(Objective-C并没有强制规定指针读写的原子性)。

但是,有了atomic来保证基本的原子性还是没有达到线程安全的,接着在调用时需要把Block先赋值给本地变量,以防止Block突然改变。因为如果不这样的话,即便是先判断了Block属性不为空,在调用之前,一旦另一个线程把Block属性设空了,程序就会crash,如下代码:

if (self.myBlock){    //此时,走到这里,self.myBlock可能被另一个线程改为空,造成crash    //注意:atomic只会确保myBlock的原子性,这种操作本身还是非线程安全的    self.myBlock(123);}

所以正确的代码是(ARC):

MyBlockType block = self.myBlock;//block现在是本地不可变的if (block){    block(123);}

在非ARC下则需要手动retain一下,否则如果属性被置空,本地变量就成了野指针了,如下代码:

//非ARCMyBlockType block = [self.myBlock retain];if (block){    block(123);}[block release];

2. 循环引用问题

循环引用是另一个使用Block时常见的问题。

在ARC下,由于__block抓取的变量一样会被Block retain,所以必须用弱引用才可以解决循环引用问题,iOS 5之后可以直接使用__weak,之前则只能使用__unsafe_unretained了,__unsafe_unretained缺点是指针释放后自己不会置空。示例代码:

//iOS 5之前可以用__unsafe_unretained//__unsafe_unretained typeof(self) weakSelf = self;__weak typeof(self) weakSelf = self;self.myBlock = ^(int paramInt){    //使用weakSelf访问self成员    [weakSelf anotherFunc];};

在非ARC下,显然无法使用弱引用,这里就可以直接使用__block来修饰变量,它不会被Block所retain的,参考代码:

//非ARC__block typeof(self) weakSelf = self;self.myBlock = ^(int paramInt){    //使用weakSelf访问self成员    [weakSelf anotherFunc];};
时间: 2024-08-05 14:04:15

block作为类的属性时用copy的相关文章

用property声明属性时,strong,copy,weak的一般用法

//一般情况下,在声明字符串对象的时候使用 copy //UI对象使用weak //其他对象使用strong @property(nonatomic,copy)NSString *name; @property(nonatomic,weak)UIImage *image; @property(nonatomic,strong)NSArray *array;

[有码有真相]python类私有属性等要点理解及测试示例代码

# encoding: utf-8 ''' 1. python约定类定义中"__"开头(至多一个"_"结尾)的属性为"相对"的私有属性 2. 私有属性在类及其子类定义中是"可见"的,在 其它范围正常访问是相对"不可见"的 3. 留下子类可以修改父类私有属性的口子方便debug测试等 4. 在类定义以外可以通过['_'+className+'私有属性名称']名称访问 5. 私有属性引入是避免类,实例等混淆和冲

IOS 类的属性修饰符atomic

在声明一个类的属性时,默认这个属性会被修饰atomic,意思是原子性访问的. nonatomic和atomic修饰的属性,在自己没有重写setter和getter的时候才会发生作用,其主要的作用可以理解为,atomic修饰的setter和getter在多线程下面是互斥的,相当于setter和getter中开始和结束的地方加锁了. stackoverflow上面对atomic解释赞同最多的答案: atomic不等于线程安全,对于常见的引用.指针.4字节(位宽字节)的变量,其赋值操作本身就是原子的

提取实体类对象和实体类 提取属性 提取关系

提取实体类和实体对象 设计者可以依据用例文本描述中出现的名词和名词短语来提取实体对象[abbott ,1983] 另一种方法是根据预先定义的概念型列表,逐项判断系统中是否有对应的实体对象,这种方法可以作为第一种方法的补充 对于一般的软件系统,一般可以从如下的概念出发寻找系统中的实体对象: 物理的或者实在的对象 人的角色 组织 时间 地点 规格说明或事务的描述 规则和策略 找到实体类后可以按照对象共性的多少对它们进行分组,以抽象出分析模型中的实体类,但是在分析阶段很难决定某一个实体对象是应该和其他

Method Resolve Order (MRO) - 类对象属性的解析顺序

Method Resolve Order (MRO) - 类对象属性的解析顺序 Python 支持多重继承, 此时就需要解决按照何种顺序来解析属性的问题.类的继承关系在一个特殊的类属性中指定(__mro__).这个属性的作用是按照序列出类及其超类, python 会按照这个顺利搜索方法.任何实现了多重继承的语言都要处理潜在的命名冲突问题,这种冲突由不相关的父类实现的同名方法所引起的,称为'菱形问题'. 举个例子, 1 class A(list): 2 def disp(self): 3 prin

OC中NSString和block的属性时copy,而不用retain

OC中NSString和block的属性时copy,而不用retain   而容器类使用retain NSString 为什么要使用copy,而不是retain 1.首先如果使用retain,只是引用计数+1,并没有生成新的对象,所以效率好 2.但是使用copy安全.因为NSString为NSMutableString 的基类,如果将NSMutableString 以retain的形式赋值给NSString后,后续修改NSMutableString会导致NSString内容的变化,这通常不是我们

Objc中为何某些类的属性要设置为copy而不是strong?

大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) 不知道大家是否注意,我们再使用一些第三方类的时候大多数情况下对象属性的type都是strong,但是某些类却设置成了copy类型,这是为什么? 看似很细小的改变其实是有其用意的. copy和strong之间的区别很清楚,前者只是直接引用一个对象,而后者却是引用一个对象的拷贝. 比如如下示例: #import <UIKit/UIKit.h> @interfac

一个类在另一个类中以集合形式声明为属性时应该注意的问题

标题:一个类在另一个类中以集合形式声明为属性时应该注意的问题,先解释一下,看下面的代码吧 [Serializable] public class MO_T_sj_quest { public List<MO_T_sj_answer> AnswerList { get; set; } public List<MO_T_sj_answer> IDList { get; set; } public List<MO_T_sj_quest_bind> BindList { get

Python类的组合时产生的属性同步改变问题

问题引入: class A: x = 1 class B: t = A() b1 = B() b2 = B() b1.t.x = 2 print(b2.t.x) >>> 2 为什么操作对象b1 会改变另一个对象 b2 的属性值?对象之间的属性应该相互独立?难道类的对象属性和列表一样是一种可变属性,操作对象属性会改变类属性,从而改变了另一个该类实例中的属性? print(A.x)>>> 1 发现类属性并没有被改变 查阅多方资料后,发现这种情况和实例的创建规则有关 clas