iOS 小技巧

IOS属性的作用

2016-02-26  本文已影响1281人  桃花流水鳜鱼肥

一、类和对象。


说明:在学习属性前先理清属性跟类和对象的关系。所以先说明一下类和对象。

类:

类是抽象概念,用于抽象描述具有相同属性和行为的某一类对象。比如:人、车等。

对象:

对象是类的实例化,具体描述这个类别中的一个实体。

类是抽象的,不占内存;对象是具体的,占内存。

举例:
“人”是一个类,人的属性有名字、身高、性别、肤色、国籍等等。人的行为有吃饭、睡觉、尿尿等等。但是,在类别里这些都是抽象的,是用来描述“人”这一类对象。
小红是一个对象,她的名字是小红、性别是女,国籍是中国。她的行为是正在睡觉。这些都是具体存在的,这些具体存在描述了小红这个对象

二、属性


a.引入属性的原因:

编译器遇到关键字@property会自动为这个属性添加setter/getter方法,避免了每次手动添加的麻烦。

b.属性与对象、类的关系。

关系:属性是一个类中用来描述对象的抽象概念,一个类可以有很多属性,一个属性可以描述对象的一个特征。

c.属性、实例变量、成员变量之间的关系。

以下面的属性代码举例:

@property (copy, nonatomic) NSString *myTitle;//声明属性

表面关系:
当我们声明一个属性的时候,编译器就会为我们自动生成对应的实例变量_myTitle,当然,我们也可以通过关键字synthesize手动生成对应的实例变量myTitle;

实质原因:链接
首先要知道属性是OC的一种新的语言机制,声明的属性必须要有相对应的实例变量。

***.h***
@interface ViewController : UIViewController
{
    //属性的实例变量
    NSString *myTitle;
}
//编译器遇到@property会自动声明对应的setter/getter
@property (copy, nonatomic) NSString *myTitle;
@end

***.m***
//编译器遇到@synthesize会自动实现setter/getter方法
//编译器遇到@synthesize回去访问myTitle的同名变量,如果没找到就报错。
@synthesize myTitle;

属性改变步骤:
1.第一版,最早版本

***.h***
@interface ViewController : UIViewController
{
    NSString *myTitle;
}
//编译器遇到@property会自动声明对应的setter/getter
@property (copy, nonatomic) NSString *myTitle;
@end

***.m***
//编译器遇到@synthesize会自动实现setter/getter方法
//编译器遇到@synthesize回去访问myTitle的同名变量,如果没找到就报错。
@synthesize myTitle;

2.第二版,不需要自己手动声明一个成员变量,关键字@synthesize会默认去访问myTitle的同名变量,如果找不到则会自动生成一个

***.h***
@interface ViewController : UIViewController
@property (copy, nonatomic) NSString *myTitle;
@end

***.m***
/*
 1.编译器遇到@synthesize会自动实现setter/getter方法;
 2.编译器遇到@synthesize回去访问myTitle的同名变量;
 3.如果找不到同名的变量,会自动生成一个私有同名变量myTitle
 4.因此现在开始就不用再手动声明一个实例变量了,
 */
@synthesize myTitle;

3.第三版,考虑到变量名和getter的方法名、setter的参数名一样,容易让人误会,引起警告,所以默认给属性生成的同名变量名带下划线。

/*
 1.指定_myTitle作为myTitle的实例变量
 2.这样我们就可以去使用_myTitle,避免和setter名同名产生误会了
 */
@synthesize myTitle = _myTitle;

4.第四版,在IOS4.5后,也@synthesize可以省略了,最终就变成这样了

@property (copy, nonatomic) NSString *myTitle;

如果我们把上面的过程全部显示出来的话,本质如下:

**.h**
#import <UIKit/UIKit.h>

@interface ViewController : UIViewController
{
/*
***被隐藏的代码:***
1.这个默认是@synthesize myTitle = _myTitle;生成的
2.所以如果我们手动设置@synthesize myTitle,那么我们编译器
  生成的变量就是NSString *myTitle,相当于@synthesize myTitle = myTitle,
  如果设置@synthesize myTitle = youTitle,那么编译器生成的变量就是NSString *youTitle了
  这要注意。
*/
    NSString *_myTitle;
}
@property (copy, nonatomic) NSString *myTitle;
//***被隐藏的代码***
//编译器遇到@property会自动声明setter/getter方法
- (void)setMyTitle:(NSString *)myTitle;
- (NSString *)myTitle;
@end

**.m**
/*
***被隐藏的代码***
1.@synthesize关键字会自动实现setter/getter的方法
2.@synthesize myTitle = _myTitle指明了属性myTitle的实例变
量是_myTitle,setter/getter操作的对象就是_myTitle.

*/
@synthesize myTitle = _myTitle;
- (void)viewDidLoad {
    [super viewDidLoad];
    _myTitle = @"123";

}

//***被隐藏的代码***
//由关键字@synthesize自动实现
- (NSString *)myTitle{
    return _myTitle;
}
- (void)setMyTitle:(NSString *)myTitle{
    _myTitle = myTitle;
}

代码说明:
1.编译器遇到关键字@property,自动声明setter/getter方法。
2.编译器遇到@synthesize,自动实现setter/getter方法。
3.@synthesize myTitle = _myTitle;为属性myTitle生成了一个实例变量_myTitle,所以我们对属性的操作self.myTitle实质上都是在操作_myTitle变量。

所以属性、实例变量和成员变量的关系是:
声明属性的时候编译器自动生成的实例变量,实例变量的本质就是成员变量,self.myTitle操作属性的时候实质上是在操作成员变量_myTitle(也就是实例变量)。


同时重写setter/getter的问题:
我们会发现,当我们同时重写setter/getter时会报错,为什么呢?这是因为当我们同时重写setter/getter时,编译器自动添加的代码@synthesize myTitle = _myTitle;失效,就不会自动为我们生成实例变量_myTitle了,setter/getter操作的对象就不存在了。所以我们要加上@synthesize myTitle = _myTitle;,手动指定setter/getter要操作的实例对象是_my

上一篇下一篇

猜你喜欢

热点阅读