编译器特性ARC

ARC(是编译器特性)

  • ARC是自iOS 5之后增加的新特性,完全消除了手动管理内存的烦琐,编译器会自动在适当的地方插入适当的retain、release、autorelease语句。你不再需要担心内存管理,因为编译器为你处理了一切
  • ARC 是编译器特性,而不是 iOS 运行时特性,它也不是类似于其它语言中的垃圾收集器。因此 ARC 和手动内存管理性能是一样的,有时还能更加快速,因为编译器还可以执行某些优化

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

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

强指针和弱指针

  • 默认所有实例变量和局部变量都是Strong指针
  • 弱指针指向的对象被回收后,弱指针会自动变为nil指针,不会引发野指针错误

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

/*文件名:Dog.h */#import <Foundation/Foundation.h>@interface Dog : [email protected]/*文件名: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 方式)xcode6

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

如何查看项目是否是 ARC?

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

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

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

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

-fno-objc-arc

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

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

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

-f-objc-arc

ARC使用注意

  • 不能调用release、retain、autorelease、retainCount
  • 可以重写dealloc,但是不能调用[super dealloc]
  • @property : 想长期拥有某个对象,应该用strong,其他对象用weak
  • 其他基本数据类型依然用assign
  • 两端互相引用时,一端用strong、一端用weak

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

/*文件名: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;
}

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

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

内存布局:

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

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

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

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

时间: 2024-08-04 12:19:36

编译器特性ARC的相关文章

Xcode编译器特性:ARC(Automatic Reference Counting)

一.         基本简介 ARC是自iOS 5/Mac OS X 10.7之后增加的新特性,消除了原先手动管理内存的烦琐,编译器会自动在适当的地方插入适当的retain.release.autorelease语句.你不再需要担心内存管理,因为编译器为你处理了一切 ARC 是编译器特性,而不是 iOS 运行时特性,它也不是类似于其它语言中的垃圾收集器.因此 ARC 和手动内存管理性能是一样的,有时还能更加快速,因为编译器还可以执行某些优化 简单地理解ARC,就是通过指定的语法,让编译器(LL

使用xmake检测编译器特性支持

如果我们要写跨平台的c/c++代码,很多时候需要处理由于不同编译器对c/c++各个标准支持力度不同导致的兼容性问题,一般通常的解决办法是:自己在代码中通过宏去判断各个编译器的版本.内置宏.标准库宏.__has_feature等来检测处理. 自己如果在代码中按上述的方式检测,会很繁琐,尤其是像c++这种存在大量语法特性,如果一一检测过来,工作量是非常大的. 通过构建工具预先检测编译特性 另外比较省事的方式,就是依赖构建工具提前做好检测,然后把检测结果作为宏添加到编译中去,这样代码只需要判断对应的特

黑马程序员---Objective-C基础学习---编译器特性@property和@synthesize

------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 编译器特性@property和@synthesize 1.@property @property可以自动生成某个成员变量的setter和getter声明. 新建一个项目,添加Person类. Person.h // // Person.h // zijia // // Created by zou on 5/10/15. // Copyright (c) 2015 __MyCompanyNam

&lt;&lt;黑马程序员&gt;&gt;OC编译器特性

点语法 特性:点语法的本质就是方法调用 成员变量的作用域 ① public :在任何地方都可以访问成员变量 在函数中可通过变量名来访问成员变量 ② private:表示私有,只能在当前类的对象方法中直接访问成员变量 ( @implemnetation中默认是@private(子类中不可访问)) ③ protected :可以在当前类和子类及其子类的对象方法中直接访问 ( 默认就是protected在interface) ④ package:只要处在同一个框架中,就能直接访问对象的变量. prop

Objective-C:08_内存管理_ARC编译器特性

自动生成代码,不是垃圾回收 ARC的判断准则:         只要没有强指针指向对象,就会释放对象   1.ARC特点     ->不允许调用release.retain.retainCount     ->允许重写dealloc,但是不允许调用[super dealloc]     ->@property的参数             strong:成员变量是强指针(适用于OC对象类型)             weak  :成员变量是弱指针(适用于OC对象类型)          

使用Xcode Instruments Leak解决内存泄漏问题

iOS 5.0之后apple引入了Xcode编译器特性ARC(Automatic Reference Counting,自动引用计数)来帮助开发者管理内存,但为了追求app的高性能与减少安装包大小,工作中很多时候需要我们手动管理内存.再牛的开发者也不能保证自己写的code 100%没有内存泄露,出现内存泄露不可怕,可怕的是我们时间与精力花了大把,但内存泄露依旧没解决,即影响了工作效率也影响自己的心情. 下面就讲解xcode中的内存调试神器---Instruments Leak ,不管是ios开发

检测内存泄漏

iOS 5.0之后apple引入了Xcode编译器特性ARC(Automatic Reference Counting,自动引用计数)来帮助开发者管理内存,但为了追求app的高性能与减少安装包大小,工作中很多时候需要我们手动管理内存.再牛的开发者也不能保证自己写的code 100%没有内存泄露,出现内存泄露不可怕,可怕的是我们时间与精力花了大把,但内存泄露依旧没解决,即影响了工作效率也影响自己的心情. 下面就讲解xcode中的内存调试神器---Instruments Leak ,不管是ios开发

IOS 开发笔记-基础 UI(8)控件连线的 strong 和 weak 设定

回忆 oc 的内存管理: objective-c 语法快速过(6)内存管理原理,objective-c 语法快速过(7)编译器特性ARC ARC是苹果为了简化程序员对内存的管理,推出的一套内存管理机制,使用ARC机制,对象的申请和释放工作会在运行时,由编译器自动在代码中添加retain和release 1> strong:强指针引用的对象,在生命周期内不会被系统释放,在OC中,对象默认都是强指针 2> weak:弱指针引用的对象,系统会立即释放,弱指针可以指向其他已经被强指针引用的对象 他们都

MRC与ARC混合开发配置

iOS4.1后,默认为ARC模式,ARC实质上通过编译器特性实现的,只需简单修改下文件配置即可实现. 1.要切换至MRC需要选中当前Project,选择Build Settings,查询框内键入auto,查找到Objective-C Automatic Reference Counting ,将YES选成NO即可.见下图: 2.还可以灵活选择单一或几个文件设置成ARC模式,方法是选择targets,找到Compile Sources,选择所需文件,将flags ,键入-fobjc-arc即可.见