Objective-C 【This is ARC】

———————————————————————————————————————————

ARC的概念及原理



(1)指针分类



强指针:默认情况下,所有的指针都是强指针,关键字strong



弱指针:_ _weak 关键字修饰的指针。



声明一个弱指针如下:



_ _weak Person *p;  (注意:这里的weak前面是两个下划线,且两个下划线之间没有空格!)





(2)ARC



Automatic Reference Counting,自动引用计数。



在工程中十分的简单,和往常一样编写代码就好,只不过永远不写retain、release和autorelease就好!这也是ARC基本原则。





(3)ARC工作原理及判断准则



工作原理:   ARC是OC编译器的特性,而不是运行时特性或者垃圾回收机制,ARC所做的只不过是在代码编译时为你自动在合适的位置插入release或者autorelease。



判断准则:   (只要没有强指针指向对象,对象就会被释放)

当使用ARC的时候,我们要暂时忘记“引用计数器”,因为判断标准变了。(标准不再是retainCount是否为0,而是有没有强指针指向)







★注意:



对于强指针指向的对象,一旦强指针指向了别处,那么对象内存空间就被释放。



如果对象同时被一个强指针和一个弱指针指向,强指针一旦指向别处,那么对象内存空间也会释放,而指向该对象的弱指针被赋值nil。(这句话告诉我们,强指针指向对象并持有该对象,而弱指针指向对象却不持有该对象)





代码:



#import <Foundation/Foundation.h>



@interface Car : NSObject

-(void)run;

@end



@implementation Car

-(void)run

{

    NSLog(@"Car run!");

}



- (void)dealloc

{

    NSLog(@"Car dealloc!");

}

@end



int main(int argc, const char * argv[]) {

    @autoreleasepool {

//        在iOS5以后,默认的都是ARC机制

//        ARC不能使用release、autorelease、retainCount和retain

//        ARC写允许重写dealloc方法,但是不能调用 [super dealloc];

        

        Car *car1=[[Car alloc]init];

        [car1 run];

    }//执行完这句话car1的内存空间释放

    NSLog(@"123456789");

    return 0;

}





———————————————————————————————————————————

ARC下 单对象的内存管理



代码:



#import <Foundation/Foundation.h>



@interface Car : NSObject

@property int speed;

-(void)run;

@end



@implementation Car

-(void)run

{

    NSLog(@"Car run!");

}



- (void)dealloc

{

    NSLog(@"Car dealloc!");

}

@end



int main(int argc, const char * argv[]) {

    @autoreleasepool {

//        Car *car=[[Car alloc]init];

//        

//        car=nil;

////        car在指向nil的时候是会被立即释放的,因为car被赋值nil,原来在堆区中car指向的内存空间没有了强指针的指向,所以内存释放。

////       因为car被设置为了nil,所以下面两行相当于是对nil进行的操作 nil.speed=12; ,但是对nil的操作是可行的却没变化的,所以输出的值car.speed还是0

//        car.speed=12;

//        NSLog(@"car.speed=%d",car.speed);

        

        

//        car1 和 car2 指向的是同一个内存空间

//        __strong Car *car1=[[Car alloc]init];//强指针也是可以加上 __strong 来显性修饰的

//        

//        __weak Car *car2=car1;

//        

//        car2=nil;//弱指针car2被赋值为nil,但是强指针car1仍指向原内存空间,所以说在执行完 car2=nil; 之后,原内存空间并不会被释放。

//        

//        car1=nil;//强指针car1被赋值为nil,那么原内存空间会在语句 car1=nil; 执行完后立即释放

//        

//        NSLog(@"**********");







//        注意:

//        ①:

//        __weak Car *car3=[[Car alloc]init];

//        __strong Car *car4=car3;

//        如果是这种写法,一开始在声明car3的时候就没有强指针指向,那么声明完car3之后内存立即释放(这句话也是完全没有意义的!)

        

//        ②:

//        __strong Car *car5=[[Car alloc]init];

//        __strong Car *car6=car5;

        

//        car5=nil;

//        NSLog(@"***********");

//        如果是两个强指针指向同一个内存区域,car5赋值nil之后,car6还指向原内存地址,所以内存并没有释放,而是在自动释放池大括号结束时释放        

    }

    return 0;

}





———————————————————————————————————————————

ARC下 多对象内存管理



这个地方没有什么好说的,如果在程序中一个实例对象中的实例变量是另一个类型的实例对象,那么我们应该将他设置成强指针类型,因为我们要用到他,用它的时候不能让他成为nil。



设置变量的时候我们可以:



@property (nonatomic,strong) Dog *dog;  //当然这里是默认的



还有就是用@property增强来设置实例变量的时候

(下面等同)

strong —————retain

weak—————assign





———————————————————————————————————————————

ARC下 循环引用问题



代码:



Dog.h



#import <Foundation/Foundation.h>

@class Person;



@interface Dog : NSObject

@property(nonatomic,strong)Person *owner;

@end





Dog.m



#import "Dog.h"



@implementation Dog

- (void)dealloc

{

    NSLog(@"Dog dealloc!");

}

@end





Person.h



#import <Foundation/Foundation.h>

@class Dog;



@interface Person : NSObject

@property(nonatomic,weak)Dog *dog;

@end





Person.m



#import "Person.h"



@implementation Person

- (void)dealloc

{

    NSLog(@"Person dealloc!");

}

@end





main.m



#import <Foundation/Foundation.h>

#import "Person.h"

#import "Dog.h"



int main(int argc, const char * argv[]) {

    @autoreleasepool {

        Person *p=[[Person alloc]init];

        

        Dog *dog=[[Dog alloc]init];

        

//        不加下面两句话的时候(没有互相建立关系的时候),他们是可以释放的。但是他们之间一旦有了关系,就无法被释放(两个都是strong类型)

        p.dog=dog;

        

        dog.owner=p;

//        想要解决这个问题,其实和前面在MRC机制下差不多,我们只需要将他们两个其中一个的类型设置为weak,这样就可以解决循环调用的问题了。

//        比如我们将Person 中的Dog类型的参数设置为weak,那么他就会先释放,然后dog的owner参数也会释放,这样两个都能释放了

    }

    return 0;

}





附加:ARC下 的 @property 参数



ARC中的 @property



strong:用于OC对象,相当于MRC中的retain

weak:用于OC对象,相当于MRC中的assign(在定义@property类型的时候就不要加两个下划线了)

assign:用于基本数据类型,跟MRC中的assign一样

copy:用于NSString,跟MRC中的copy一样



★在ARC下解决循环引用问题,一边用strong,一边用weak





———————————————————————————————————————————

ARC的兼容和转换

①如何在ARC文件中兼容非ARC的文件

Objective-C 【This is ARC】

步骤就在上图了,Dog类是ARC,Car类是非ARC。那么我们就找到Car类的Compiler Flags,然后双击在里面键入 -fno-objc-arc  如下图(如果是-f-objc-arc就是兼容ARC)



Objective-C 【This is ARC】





②如何将MRC文件转换成ARC文件

首先确保你要转换的文件之前是一个MRC文件,也就是将ARC的开关关闭

Objective-C 【This is ARC】

然后按照下图的操作,先选定该target,然后Edit,然后Convert,然后To Objective-C ARC…

Objective-C 【This is ARC】

最后按照提示下一步,你就会发现,你的代码发生了一些改变:

Objective-C 【This is ARC】

这就是将MRC文件转换成ARC文件的方法。但是,注意,这种方法不一定100%成功,可能会转换错误,所以用的时候要谨慎!





———————————————————————————————————————————

版权声明:本文为博主原创文章,未经博主允许不得转载。

上一篇:PLSQL DEVELOPER编辑器的自动替换文件


下一篇:转-4年!我对OpenStack运维架构的总结