成员变量和属性

来源:互联网 发布:淘宝抢购软件 编辑:程序博客网 时间:2024/06/05 22:54

http://blog.sina.com.cn/s/blog_9075354e0102uwwt.html

一、类Class中的属性property

ios第一版中,我们为输出口同时声明了属性和底层实例变量,那时,属性是oc语言的一个新的机制,并且要求你必须声明与之对应的实例变量,例如:


@interface MyViewController :UIViewController

{

    UIButton *myButton;

}

@property (nonatomic, retain) UIButton *myButton;

@end


最近,苹果将默认编译器从GCC转换为LLVM(low level virtual machine),从此不再需要为属性声明实例变量了。如果LLVM发现一个没有匹配实例变量的属性,它将自动创建一个以下划线开头的实例变量。因此,在这个版本中,我们不再为输出口声明实例变量。


例如:MyViewController.h文件

@interface MyViewController :UIViewController

@property (nonatomic, retain) UIButton *myButton;

@end


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

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

例如在oc中有如下代码

 .h文件

@interface MyViewController :UIViewController

{

    NSString *name;

}

@end


 .m文件中,self.name这样的表达式是错误的。xcode会提示你使用->,改成self->name就可以了。因为oc中点表达式是表示调用方法,而上面的代码中没有name这个方法。

  oc语法关于点表达式的说明:"点表达式(.)看起来与C语言中的结构体访问以及java语言汇总的对象访问有点类似,其实这是oc的设计人员有意为之。如果点表达式出现在等号 左边,该属性名称的setter方法将被调用。如果点表达式出现在右边,该属性名称的getter方法将被调用。"

  所以在oc中点表达式其实就是调用对象的settergetter方法的一种快捷方式例如:dealie.blah = greeble 完全等价于 [dealie.blah setBlah:greeble];


 以前的用法,声明属性跟与之对应的实例变量:


@interface MyViewController :UIViewControlle

{

    UIButton *myButton;

}

@property (nonatomic, retain) UIButton *myButton;

@end


  这种方法基本上使用最多,现在大部分也是在使用,因为很多开源的代码都是这种方式。但是ios5更新之后,苹果是建议以以下的方式来使用:


@interface MyViewController :UIViewController

@property (nonatomic, retain) UIButton *myButton;

@end


  因为编译器会自动为你生成以下划线开头的实例变量_myButton,不需要自己手动再去写实例变量。而且也不需要在.m文件中写@synthesize myButton;也会自动为你生成settergetter方法。@synthesize的作用就是让编译器为你自动生成settergetter方法。

  @synthesize 还有一个作用,可以指定与属性对应的实例变量,例如@synthesize myButton = xxx;那么self.myButton其实是操作的实例变量xxx,而不是_myButton了。

  在实际的项目中,我们一般这么写.m文件


@synthesize myButton;


  这样写了之后,那么编译器会自动生成myButton的实例变量,以及相应的gettersetter方法。注意:_myButton这个实例变量是不存在的,因为自动生成的实例变量为myButton而不是_myButton所以现在@synthesize的作用就相当于指定实例变量;

  如果.m文件中写了@synthesize myButton;那么生成的实例变量就是myButton;如果没写@synthesize myButton;那么生成的实例变量就是_myButton。所以跟以前的用法还是有点细微的区别。


 二、类别中的属性property

  类与类别中添加的属性要区分开来,因为类别中只能添加方法,不能添加实例变量。经常会在ios的代码中看到在类别中添加属性,这种情况下,是不会自动生成实例变量的。比如在:UINavigationController.h文件中会对UIViewController类进行扩展

@interface UIViewController (UINavigationControllerItem)

@property(nonatomic,readonly,retain) UINavigationItem *navigationItem;

@property(nonatomic) BOOL hidesBottomBarWhenPushed;

@property(nonatomic,readonly,retain) UINavigationController *navigationController;

@end

  这里添加的属性,不会自动生成实例变量,这里添加的属性其实是添加的gettersetter方法。

  注意一点,匿名类别(匿名扩展)是可以添加实例变量的,非匿名类别是不能添加实例变量的,只能添加方法,或者属性(其实也是方法)。



@synthesize@dynamic区别

在声明property属性后,有2种实现选择

@synthesize

编译器期间,让编译器自动生成getter/setter方法。

当有自定义的存或取方法时,会自动屏蔽自动生成的该方法

 

@dynamic告诉编译器,不自动生成getter/setter方法,避免编译期间产生警告

然后由自己实现存取方法

或存取方法在运行时动态创建绑定:主要使用在CoreData的实现NSManagedObject子类时使用,由Core Data框架在程序运行的时动态生成子类属性


@dynamic 意思是由开发人员提供相应的代码:对于只读属性需要提供 getter,对于读写属性需要提供 gettersetter

@synthesize 意思是,除非开发人员已经做了,否则由编译器生成相应的代码,以满足属性声明。

0 0
原创粉丝点击