博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Objective-C中的@property和@synthesize用法
阅读量:5316 次
发布时间:2019-06-14

本文共 3577 字,大约阅读时间需要 11 分钟。

@代表“Objective-C”的标志,证明您正在使用Objective-C语言

 

Objective-C语言关键词,@property与@synthesize配对使用。

 

功能:让编译好器自动编写一个与数据成员同名的方法声明来省去读写方法的声明。

 

如:

1、在头文件中:

C代码  
  1. @property int count;  

等效于在头文件中声明2个方法:

C代码  
  1. - (int)count;  
  2. -(void)setCount:(int)newCount;  

 

2、实现文件(.m)中

C代码  
  1. @synthesize count;  

等效于在实现文件(.m)中实现2个方法。

C代码  
  1. - (int)count  
  2. {  
  3.     return count;  
  4. }  
  5. -(void)setCount:(int)newCount  
  6. {  
  7.     count = newCount;  
  8. }  

  

以上等效的函数部分由编译器自动帮开发者填充完成,简化了编码输入工作量。

 

格式:

 

声明property的语法为:@property (参数1,参数2) 类型 名字;

 

如:

C代码  
  1. @property(nonatomic,retain) UIWindow *window;  

 

其中参数主要分为三类:

 

读写属性: (readwrite/readonly)

setter语意:(assign/retain/copy)

原子性: (atomicity/nonatomic)

 

各参数意义如下:

 

readwrite: 产生setter\getter方法

readonly: 只产生简单的getter,没有setter。

assign: 默认类型,setter方法直接赋值,而不进行retain操作

retain: setter方法对参数进行release旧值,再retain新值。

copy: setter方法进行Copy操作,与retain一样

nonatomic: 禁止多线程,变量保护,提高性能

 

参数类型

参数中比较复杂的是retain和copy,具体分析如下:

 

getter 分析

1、

C代码  
  1. @property(nonatomic,retain)test* thetest;  
  2. @property(nonatomic ,copy)test* thetest;  

等效代码:

C代码  
  1. -(void)thetest  
  2. {  
  3.   return thetest;  
  4. }  

2、

C代码  
  1. @property(retain)test* thetest;  
  2. @property(copy)test* thetest;  

等效代码:

C代码  
  1. -(void)thetest  
  2. {  
  3.     [thetest retain];  
  4.     return [thetest autorelease];  
  5. }  

 

setter分析

 

1、

C代码  
  1. @property(nonatomic,retain)test* thetest;  
  2. @property(retain)test* thetest;  

等效于:

C代码  
  1. -(void)setThetest:(test *)newThetest {  
  2.     if (thetest!= newThetest) {  
  3.         [thetestrelease];  
  4.         thetest= [newThetest retain];  
  5.     }  
  6. }  

  

 2、

C代码  
  1. @property(nonatomic,copy)test* thetest;  
  2. @property(copy)test* thetest;  

 等效于:

C代码  
  1. -(void)setThetest:(test *)newThetest {  
  2.     if (thetest!= newThetest) {  
  3.         [thetest release];  
  4.         thetest= [newThetest copy];  
  5.     }  
  6. }  

 

nonatomic

如果使用多线程,有时会出现两个线程互相等待对方导致锁死的情况(具体可以搜下线程方面的注意事项去了解)。在没有(nonatomic)的情况下,即默认(atomic),会防止这种线程互斥出现,但是会消耗一定的资源。所以如果不是多线程的程序,打上(nonatomic)即可

 

retain

代码说明

如果只是@property NSString*str; 则通过@synthesize自动生成的setter代码为:

C代码  
  1. -(void)setStr:(NSString*)value{  
  2.     str=value;  
  3. }  

  

如果是@property(retain)NSString*str; 则自动的setter内容为:

C代码  
  1. -(void)setStr:(NSString*)v{  
  2.     if(v!=str){  
  3.         [str release];  
  4.         str=[v retain];  
  5.     }  
  6. }  

 

 

所有者属性

我们先来看看与所有权有关系的属性,关键字间的对应关系。

属性值 关键字 所有权

strong __strong
weak __weak
unsafe_unretained __unsafe_unretained
copy __strong
assign __unsafe_unretained
retain __strong

strong

该属性值对应 __strong 关键字,即该属性所声明的变量将成为对象的持有者。

weak

该属性对应 __weak 关键字,与 __weak 定义的变量一致,该属性所声明的变量将没有对象的所有权,并且当对象被破弃之后,对象将被自动赋值nil。

并且,delegate 和 Outlet 应该用 weak 属性来声明。同时,如上一回介绍的 iOS 5 之前的版本是没有 __weak 关键字的,所以 weak 属性是不能使用的。这种情况我们使用 unsafe_unretained。

unsafe_unretained

等效于__unsafe_unretaind关键字声明的变量;像上面说明的,iOS 5之前的系统用该属性代替 weak 来使用。

copy

与 strong 的区别是声明变量是拷贝对象的持有者。

assign

一般Scalar Varible用该属性声明,比如,int, BOOL。

retain

该属性与 strong 一致;只是可读性更强一些。 

 
声明的分类

在  中的Property一章里有对类Property详细说明。

@property中的声明列表已分类为以下几种:

1, 声明属性的访问方法:

  • getter=getterName
  • setter=setterName
    声明访问属性的设置与获取方法名。

2,声明属性写操作权限:

  • readwrite
    声明此属性为读写属性,即可以访问设置方法(setter),也可以访问获取方法(getter),与readonly互斥。
  • readonly
    声明此属性为只读属性,只能访问此属性对应的获取方法(getter),与readwrite互斥。

3,声明写方法的实现:

  • assign
    声明在setter方法中,采用直接赋值来实现设值操作。如:
C代码  
  1. -(void)setName:(NSString*)_name{  
  2.      name = _name;  
  3. }  
 
  • retain
    声明在setter方法中,需要对设过来的值进行retain 加1操作。如:
C代码  
  1. -(void)setName:(NSString*)_name{  
  2.      //首先判断是否与旧对象一致,如果不一致进行赋值。  
  3.      //因为如果是一个对象的话,进行if内的代码会造成一个极端的情况:当此name的retain为1时,使此次的set操作让实例name提前释放,而达不到赋值目的。  
  4.      if ( name != _name){  
  5.           [name release];  
  6.           name = [_name retain];  
  7.      }  
  8. }  
 
  • copy
    调用此实例的copy方法,设置克隆后的对象。实现参考retain。

4,访问方法的原子性:

    • nonatomic
      在默认的情况下,通过synthesized 实现的 setter与getter 都是原子性访问的。多线程同时访问时,保障访问方法同时只被访问一个线程访问,如:
    • C代码  
      1. [ _internal lock ]; // lock using an object-level lock  
      2. id result = [ [ value retain ] autorelease ];  
      3. [ _internal unlock ];  
      4. return result;  
       
    • 但如果设置nonatomic时,属性的访问为非原子性访问。

转载于:https://www.cnblogs.com/ShaoYinling/p/4382575.html

你可能感兴趣的文章
@Component 和 @Bean 的区别
查看>>
Linux安全篇-iptables
查看>>
div中溢出文字用点代替
查看>>
UVA - 489 Hangman Judge
查看>>
返回一个一维数组环中的数相加的最大的和
查看>>
55分钟学会正则表达式
查看>>
python1119-20181205作业-郭恩赐提交
查看>>
LA3490 Generator(KMP + 高斯消元)
查看>>
kaldi学习 - 一脚本流学习工具使用
查看>>
vSphere SDK for Java 示例
查看>>
AOj448有趣的矩阵
查看>>
字符串比较——compareTo函数
查看>>
Eclipse使用过程中出现java.lang.NoClassDefFoundError的解决方案
查看>>
Attribute "resultClass" must be declared for element type "insert".
查看>>
字符串的排列
查看>>
[洛谷P1430]序列取数
查看>>
SQL Server数据库开发中的十大问题
查看>>
C++ string、char *、char[]、const char*
查看>>
配置WinRM的Https
查看>>
构建之法阅读笔记04
查看>>