您的位置:首页 > 移动开发 > Objective-C

黑马程序员:Objective-C 属性特性

2014-04-10 11:11 330 查看
----------- android培训java培训、java学习型技术博客、期待与您交流! ------------

assign:指定setter方法用简单的赋值,这是默认操作。你可以对标量类型(如int)使用这个属性。你可以想象一个float,它不是一个对象,所以它不能retain、copy。

retain:指定retain应该在后面的对象上调用,前一个值发送一条release消息。你可以想象一个NSString实例,它是一个对象,而且你可能想要retain它。

retain:释放旧的对象,将旧对象的值赋予输入对象,再提高输入对象的索引计数为1 ,使用retain: 对其他NSObject和其子类 ,retain,是说明该属性在赋值的时候,先release之前的值,然后再赋新值给属性,引用再加1。

 

copy:指定应该使用对象的副本(深度复制),前一个值发送一条release消息。基本上像retain,但是没有增加引用计数,是分配一块新的内存来放置它。copy是创建一个新对象,retain是创建一个指针,引用对象计数加1。copy:建立一个索引计数为1的对象,然后释放旧对象,copy是创建一个新对象,retain是创建一个指针,引用对象计数加1。

readonly:将只生成getter方法而不生成setter方法(getter方法没有get前缀)。

readwrite:默认属性,将生成不带额外参数的getter和setter方法(setter方法只有一个参数)。

atomic:对于对象的默认属性,就是setter/getter生成的方法是一个原子操作。如果有多个线程同时调用setter的话,不会出现某一个线程执行setter全部语句之前,另一个线程开始执行setter的情况,相关于方法头尾加了锁一样。

nonatomic:不保证setter/getter的原子性,多线程情况下数据可能会有问题。nonatomic,非原子性访问,不加同步,多线程并发访问会提高性能。先释放原先变量,再将新变量
     retain然后赋值;

      注意,如果不加此属性,则默认是两个访问方法都为原子型事务访问。

----------------------------------------------------------------------------------------------------------------------------

----------------------------------------------------------------------------------------------------------------------------

实际的代码演示:

@property (nonatomic, assign) int number;

这里定义了一个int类型的属性, 那么这个int是简单数据类型,本身可以认为就是原子访问,所以用nonatomic,  不需要进行引用计数,所以用assign。 适用于所有简单数据类型。

@property (nonatomic, copy) NSString *
myString;

这里定义了一个NSString类型的属性,不需要原子操作,所以用nonatomic.

为什么需要copy,而不是retain呢! 因为如果对myString赋值原字符串是一个可变的字符串(NSMutableString)对象的话,用retain的话,当原字符串改变的时候你的myString属性也会跟着变掉。我想你不希望看到这个现象。 

@property (nonatomic, retain) UIView *
myView;

这里定义了一个UIView类型的属性,不需要原子操作,所以用nonatomic.

当对myView 赋值的时候原来的UIView对象retainCount会加1

//声明文件
@interface MyClass : NSObject
@property (nonatomic, assign)   int             
number;
@property (nonatomic, copy)   NSString 
* myString;
@property (nonatomic, retain) UIView   
* myView;
@end

//实现文件
@implementation MyClass
@synthesize number;
@synthesize myString;
@synthesize myView;

//释放内存

-(void) dealloc

{

[myString release];  //copy的属性需要release;

[myView release];    //retain的属性需要release;

[super dealloc]; //传回父对象

}

@end

假如你有一段代码创建了一个MyClass对象

MyClass * instance  = [MyClass alloc] init];

//number赋值,没什么可说的, 简单数据类型就这样
instance.number = 1;

//创建一个可变字符串
NSMutableString * string = [NSMutableString stringWithString:@"hello"];

instance.myString = string;                   //对myString赋值

[string appendString:@" world!"];      //往string追加文本

NSLog(@”%@”,string);                        //此处string已经改变, 输出为 “hello world!”

NSLog(@”%@”,instance.myString);  
//输出myString,你会发现此处输出仍然为 “hello” 因为 myString在string改变之前已经copy了一份副本

UIView * view = [[UIView alloc] init];
NSLog(@”retainCount = %d”,view.retainCount);

//输出view的引用计数, 此时为1

instance.myView = view; //对myView属性赋值

NSLog(@”retainCount = %d”,view.retainCount);

//再次输出view的引用计数, 此时为2,因为myView对view进行了一次retain。

[view release];

//此处虽然view被release释放掉了,但myView对view进行了一次retain,那么myView保留的UIView的对象指针仍然有效。

[instance release] ;

----------- android培训java培训、java学习型技术博客、期待与您交流! ------------

详情请查看:http://edu.csdn.net/heima
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: