编译器特性ARC

发布时间:2020-07-12 05:16:36 作者:张涛泽
来源:网络 阅读:222

ARC(是编译器特性)

ARC基本原理(不是类似 java 的垃圾回收机制)

ARC 的规则非常简单:只要还有一个强指针变量指向对象,对象就会保持在内存中

强指针和弱指针

ARC的判断准则:只要没有强指针指向对象,就会释放对象,弱指针不会这样,及时有弱指针指向对象,对象没有强指针指向,也会自动释放掉。一般,无需显式声明为强指针,但是在封装里,定义方法的时候需要写明。而弱指针,必须显式说明。默认是强指针。

ARC特点

 1> 不允许调用release、retain、retainCount

 2> 允许重写dealloc,但是不允许调用[super dealloc]

 3> @property的参数

  * strong :成员变量是强指针(适用于OC对象类型)

  * weak :成员变量是弱指针(适用于OC对象类型)

  * assign : 适用于非OC对象类型

 4> 以前的retain改为用strong

oc的指针分2种:

 1> 强指针:默认情况下,所有的指针都是强指针 __strong

 2> 弱指针:__weak

编译器特性ARC

/*文件名:Dog.h */#import <Foundation/Foundation.h>@interface Dog : NSObject@end/*文件名:Dog.m */#import "Dog.h"@implementation Dog- (void)dealloc
{
    NSLog(@"Dog is dealloc");
}@end/*文件名:Person.h */#import <Foundation/Foundation.h>@class Dog;@interface Person : NSObject
@property (nonatomic, strong) Dog *dog;
@property (nonatomic, strong) NSString *name;
@property (nonatomic, assign) int age;@end/*文件名:Person.m */#import "Person.h"@implementation Person- (void)dealloc
{
    NSLog(@"Person is dealloc");    // [super dealloc];不能写,否则报错}@end//  main.m#import <Foundation/Foundation.h>#import "Person.h"#import "Dog.h"int main()
{
    Dog *d = [[Dog alloc] init];
    Person *p = [[Person alloc] init];
    p.dog = d;
    
    d = nil;
    NSLog(@"%@", p.dog);    
    return 0;
}void test()
{    // 错误写法(没有意义的写法)
    __weak Person *p = [[Person alloc] init];
    NSLog(@"%@", p);
    NSLog(@"------------");
}

编译器特性ARC

重构旧代码(手动内存管理重构为 ARC 方式)xcode6

编译器特性ARC

这样操作之后,可以把非 ARC 项目,转换为 ARC 项目。

如何查看项目是否是 ARC?

在 build settings 里搜索 auto,看选项:

编译器特性ARC

如何使得 ARC 和非 ARC 在一个项目共存?

经常需要使用第三方框架,或者一些其他的旧代码,那么有支持 ARC 的,也有不支持的,怎么办呢?可以这样设置:在编译选项里

编译器特性ARC

双击需要非 ARC的文件,如下设置:

-fno-objc-arc

这样这个文件就能使用 retain ,release,autorelease等关键字

-f 代表 flags 标记的意思,固定写法。

 

反过来,对于非 ARC 项目,这样设置:

-f-objc-arc

ARC使用注意

同样,在 ARC 项目里,也有循环双端引用的现象,你 strong 我,我 strong 你的情况。解决办法照旧。两端互相引用时,一端用strong、一端用weak

编译器特性ARC

/*文件名:Dog.h */#import <Foundation/Foundation.h>@class Person;@interface Dog : NSObject
@property (nonatomic, weak) Person *person;@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, strong) 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"/*
 当两端循环引用的时候,解决方案:
 1> ARC
 1端用strong,另1端用weak
 
 2> 非ARC
 1端用retain,另1端用assign */int main()
{
    Person *p = [[Person alloc] init];
    Dog *d = [[Dog alloc] init];
    p.dog = d;
    d.person = p;    return 0;
}

编译器特性ARC

否则,同样是报错,比如都使用 strong 属性

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

内存布局:

编译器特性ARC

p.dog = d;//把dog对象赋值给 person 对象里的_dog,指针,是个强指针。

d.person = p;//同样,dog 对象里的_person 强指针指向了 person 对象

编译器特性ARC

当程序执行完毕,或者说 main 函数执行完毕,自动变量销毁

编译器特性ARC

因为都是强指针,发生如上情况,内存泄露。故__weak 或者 weak 属性一般用在循环引用的场合,其他场合不多见。


推荐阅读:
  1. Swift2.0(17)内存管理ARC
  2. ARC 导入 MRC文件

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

编译器 垃圾回收 release

上一篇:C/C++/Linux运维学习之基础篇

下一篇:运维堡垒机的起源、发展和未来趋势

相关阅读

您好,登录后才能下订单哦!

密码登录
登录注册
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》