OC之属性与实例变量

目录

  1. 属性和实例变量的关系
  2. 关于强引用、弱引用
  3. 关于Copy 与 mutableCopy

属性和实例变量的关系

很多时候我们会看到下面这种写法:

@interface testClass : NSObject
{
    NSString *_token;
}
@property (nonatomic) NSString *token;
@end

之前一直不明白这种写法的意义。直到遇到了这个问题:

我在一个自定义的NSObject类里面添加了一个属性,叫做token,
然后,我想在.m文件夹里面重写他的set方法和get方法:
- (void)setToken:(NSString *)token
{
_token = token;
[[NSUserDefaults standardUserDefaults] setObject:token forKey:@"token"];
[[NSUserDefaults standardUserDefaults] synchronize];
}
- (NSString *)token
{
return [[NSUserDefaults standardUserDefaults] objectForKey:@"token"];
}

但是每次我只要两个方法都写上了,上面那行:_token = token;就会报错,如图:


.m文件

但是如果只写set方法,不写get方法了它又不会报错。
=======================
.h界面有写属性。


.h文件

这种写法的真正原因是这样的:
当你复写了get和set方法之后@property默认生成的@synthesize就不会起作用了,这也就意味着你的类不会自动生成出来实例变量了。你如果要复写set、get方法你就必须要自己声明实例变量。

@interface testClass : NSObject
{
    NSString *_token;
}
@property (nonatomic) NSString *token;
@end

关于强引用、弱引用

  • 强引用与弱引用指的是:你的指针(变量名) 与你指针指向的对象(你实例化出来的对象)的关系
  • 弱引用只是指向一个强引用的对象,并不会对对象的计数+1 (不会rerain,而是assign),所以当你的强引用对象被释放(relase)时 ,弱引用指针的指向变成了 nil
  • 弱引用主要解决的是循环引用的问题,关于循环引用出现的原因有如下情况

    delegate为什么用弱引用
    避免retain cycle,即:有A,B两个Object, A中有一个B的实例变量,B中又有一个 A的实例变量,要release A就必须releaseA中的B,而要release B有必须release B中的A,这样就产生了一个Retain Circle,A B都不能被dealloc.解决Retain Circle的方法就是使用弱引用(weak reference),弱引用没有被引用的那个Object的所有权,也就不需要release它,从而解决了Retain Circle问题.为了防止Retain Circle的发生, delegate通常都是弱引用的, 因此我们一般不应该retain一个delegate。

总结:如果两个对象之间存在互相引用则必须使用弱引用


关于Copy 与 mutableCopy

声明:这里的知识点是基于这篇文章的总结。如果你先看到我写的这篇文章,而且有让你看不懂的地方,你可以看看这篇文章,里面有实例代码。感谢AlwaysBlue 的总结。

关于非容器类的验证。 例NSString
  1. 你实例化的对象为可变字符串
NSString *NSString *str = [NSMutableString stringWithString:@"五月天的小蚂蚁"]

[str copy];
[str mutableCopy];

这种情况下,无论你使用copy 还是mutableCopy 都会创建出新的对象。

  1. 你实例化的对象是不可变字符串
NSString *str = [NSString stringWithFormat:@"%@",@"五月天的小蚂蚁"];
  • 这种情况下,当你使用

      NSString *newStr =  [str copy];
    

产生的newStr只是拷贝了原本字符串str的指针,它们指向的是同一个对象。所以这是你改变str的值,相应的,newStr的值也会改变。

  • 当你使用

      NSString *newStr = [str mutableCopy];
    

程序会复制 *str所指向的内存中的内容到一块新的内存中,并把新内存的地址赋值给*newStr. 因为是两块不相干内存区域,所以你改变一个的值,另一个不会受到影响。

非容器对象copy 与 mutableCopy 总结:

  1. 可变对象无论是copy 还是 mutableCopy 都是创建新的内存。(就是创建出了一个新的对象)
  2. 不可变对象使用 copy ,只是进行指针的复制,并没有创建出新的内存。(它们指向的是同一个对象)
  3. 不可变对象使用 mutableCopy ,创建出一块新的内存。(就是创建出了一个新的对象)
容器对象 例 NSArray

其实容器对象关于copy 与 mutableCopy 的用法结果是一样的。

所以容器对象arr如果使用 copy 那么它只是得到了原对象arr的指针,(假如我们称copy出来的对象为copyArr) 那么copyArr仍旧指向以前的内存空间,所以copyArr里面的元素仍旧是以前的元素,所以当arr改变它的元素的值时,copyArr的元素的值也相应的改变了。当然arr本身对所包含的元素进行增加删除也是会影响copyArr的。

自定义对象

未完待续.....

你可能感兴趣的:(OC之属性与实例变量)