@synthesize方法--iOS中成员变量和属性区别

@synthesize方法--iOS中成员变量和属性区别

历史由来:

接触iOS的人都知道,@property声明的属性默认会生成一个_类型的成员变量,同时也会生成setter/getter方法。 但这只是在iOS5之后,苹果推出的一个新机制。看老代码时,经常看到一个大括号里面定义了成员变量,同时用了@property声明,而且还在@implementation中使用@synthesize方法。 如下:

@interface ViewController ()

{

// 1.声明成员变量

NSString *myString;

}

// 2.在用@property

@property(nonatomic, copy) NSString *myString;

@end

@implementation ViewController

//3.最后在@implementation中用synthesize生成set方法

@synthesize myString;

@end


其实,发生这种状况根本原因是苹果将默认编译器从GCC转换为LLVM(low level virtual machine),才不再需要为属性声明实例变量了。

在没有更改之前,属性的正常写法需要成员变量+ @property + @synthesize 成员变量三个步骤。 如果我们只写成员变量+ @property:

@interface GBViewController :UIViewController

{

NSString *myString;

}

@property (nonatomic, strong) NSString *myString;

@end

编译时会报警告:

Autosynthesized property '�myString' will use synthesized instance variable '_myString', not existing instance variable 'myString'

但更换为LLVM之后,编译器在编译过程中发现没有新的实例变量后,就会生成一个下划线开头的实例变量。因此现在我们不必在声明一个实例变量。(注意:==是不必要,不是不可以==) 当然我们也熟知,@property声明的属性不仅仅默认给我们生成一个_类型的成员变量,同时也会生成setter/getter方法。

在.m文件中,编译器也会自动的生成一个实例变量_myString。那么在.m文件中可以直接的使用_myString实例变量,也可以通过属性self.myString.都是一样的。

注意这里的self.myString其实是调用的myString属性的setter/getter方法。这与C++中点的使用是有区别的,C++中的点可以直接访问成员变量(也就是实例变量)。

例如在oc中有如下代码

@interface MyViewController :UIViewController

{

NSString *name;

}

@end

在这段代码里面只是声明了一个成员变量,并没有setter/getter方法。所以访问成员变量时,可以直接访问name,也可以像C++一样用self->name来访问,但绝对不能用self.name来访问。

* 扩展:很多人觉得OC中的点语法比较奇怪,实际是OC设计人员有意为之。

* 点表达式(.)看起来与C语言中的结构体访问以及java语言汇总的对象访问有点类似,如果点表达式出现在等号 = 左边,调用该属性名称的setter方法。如果点表达式出现在=右边,调用该属性名称的getter方法。

* OC中点表达式(.)其实就是调用对象的setter和getter方法的一种快捷方式,self.myString = @"张三";实际就是[self setmyString:@"张三"];

首先我们要明白,@synthesize 生成了setter/getter方法。 虽然现在直接使用@property时,编译器会自动为你生成以下划线开头的实例变量_myString,不需要自己手动再去写实例变量。而且也不在.m文件中通过@synthesize myString;生成setter/getter方法。但在看老代码的时候,我们依旧可以看到有人使用成员变量+ @synthesize 成员变量的形式。

那么问题来了: 我们能否认为新编译器LLVM下的@property == 老编译器GCC的 成员变量+ @property + @synthesize 成员变量呢?

答案是否定的。 因为成员变量+ @property + @synthesize 成员变量的形式,编译器不会帮我们生成_成员变量,因此不会操作_成员变量了; 同时@synthesize 还有一个作用,可以指定与属性对应的实例变量, 例如@synthesize myString = xxx; 那么self.myString其实是操作的实例变量xxx,而非_String了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 文章结构 历史由来@dynamic与@synthesize的区别 一、历史由来: 接触iOS的人都知道,@prop...
    leonardni阅读 324评论 0 0
  • ——读书日断想 文 / 刘东黎 “世间数百年旧家无非积德,天下第一等好事还是读书。”这是一位清代官员自题的书房楹联...
    27dd5c406899阅读 603评论 0 0
  • 本人非技术人员,也非专业测试人员。只是从自己日常使用的角度来对比这几个笔记类app。(还有就是,既然这是自己选的题...
    大梦斋阅读 4,456评论 0 3
  • TAG 1 洗漱用品、睡衣拖鞋、DV加电脑;准考证和一本书便是全部;没有相机、没有沉重的脚架、更没有娇贵烦人的镜头...
    Zauberwurst阅读 510评论 1 1
  • 一句话说出你的目标:和谐有爱,亲密的亲密关系。 【咖啡冥想】 1今天是周一是种智慧种子的日子。虽然平时参加读书会的...
    黛儿微笑阅读 576评论 0 0