iOS arc VS mrc学习笔记

一、* Core Foundation与objective-c Object进行交换 *

对于Core Foundation与objective-cObject进行交换时,需要用到的ARC管理机制有:

(1) (__bridge_transfer) op or alternatively CFBridgingRelease(op) is used to consume a retain-count of a CFTypeRef while transferring it over to ARC. This could also be represented by

        id someObj =(__bridge <NSType>) op;
       CFRelease(op);

(2) (__bridge_retained) op or alternatively CFBridgingRetain(op) is used to hand an NSObject over to CF-land while giving it a +1 retain count. You should handle a CFTypeRef you create this way the same as you would handle a result of CFStringCreateCopy().This could also be represented by

CFRetain((__bridge CFType)op);
CFTypeRef someTypeRef =(__bridge CFType)op;

(3) __bridge just casts between pointer-land and Objective-C object-land. If you have no inclination to use the conversions above, use this one.

二、* ARC和IOS4 *

ARC在IOS4是没有 __weak 关键字的,需要使用 unsafe_unretained来代替。

三、* ARC中的内存泄露 *

使用了ARC也并不意味着我们的工程里面不会出现内存泄露了。在ARC机制下,最常见导致内存泄露的是循环强引用。容易出现的场合有:

①Outlet类型指针

Outlet类型的指针变量应该用weak属性来声明

②委托

一定要将delegate的属性设为weak,原因我就不解释了,实在不明白,请猛击这里

③block

下面这段代码,在MRC条件下是没有问题的:

MyViewController * __block myController = [[MyViewController alloc] init…];
// ...
myController.completionHandler =  ^(NSInteger result) {
    [myController dismissViewControllerAnimated:YES completion:nil];
}; 

但是在ARC条件下,就会内存泄露,导致myController指向的对象无法释放。

原因是,__block id x声明的变量x用于block中时,MRC条件下是不会增加x的引用计数,但是在ARC条件下,会使x得引用计数加一,请各位务必注意!!!!!!!!!!!!

上述问题代码有以下几种解决方案:

方案一:

MyViewController * __block myController = [[MyViewController alloc] init…];
// ...
myController.completionHandler =  ^(NSInteger result) {
    [myController dismissViewControllerAnimated:YES completion:nil];
    myController = nil;
}; 

最简单的解决办法,在block中使用完myController时,是它指向nil,没有strong类型的指针指向myController指向的对象时,该对象就回被释放掉。

方案二:

MyViewController *myController = [[MyViewController alloc] init…];
// ...
MyViewController * __weak weakMyViewController = myController;
myController.completionHandler =  ^(NSInteger result) {
    [weakMyViewController dismissViewControllerAnimated:YES completion:nil];
};  

该方案使用了一个临时的__weak类型的指针weakMyViewController,在block中使用该指针不会导致引用计数加一,但却存在隐患,当该对象在外部被释放时,block里面执行的操作就无效了。下面的方案三可以解决这个问题。

方案三:

MyViewController *myController = [[MyViewController alloc] init…];
// ...
MyViewController * __weak weakMyController = myController;
myController.completionHandler =  ^(NSInteger result) {
    MyViewController *strongMyController = weakMyController;
    if (strongMyController) {
        // ...
        [strongMyController dismissViewControllerAnimated:YES completion:nil];
        // ...
    }
    else {
        // Probably nothing...
    }
};  

即在block中使用myController对象之前再声明一个临时的strong类型的指针,指向weak类型的指针,这时strongMyController指针就变成了有效的强引用,其指向的对象就能保证不被释放掉。

④定时器

定时器也是非常容易产生内存泄露的地方。比如下面的代码

@implementation AnimatedView
{
  ?NSTimer *timer;
?}  

- (id)initWithCoder:(NSCoder *)aDecoder
{
?
?  if ((self = [super initWithCoder:aDecoder])){
    timer = [NSTimer scheduledTimerWithT    imeInterval:0.1
    target:self
    selector:@selector(handleTimer:)
    userInfo:nil
    repeats:YES];
  }
  return self;
}
- (void)dealloc
{
  [timer invalidate];
}
- (void)handleTimer:(NSTimer*)timer
{
  //do something
}  

乍一看这段代码没啥问题,但是运行起来才发现dealloc方法是不会被调用的,self有一个timer的强引用,timer又有一个self的强引用,典型的循环引用!

解决方法是将timer的属性设置为__weak。

四、* @autoreleasepool 和 NSAutoreleasePool *

ARC中是不支持使用NSAutoreleasePool的,但是可以使用@autoreleasepool代替。@autoreleasepool既可以用在ARC环境中,也可以用在非ARC环境中,而且效率要比前者高,苹果官网中是这样描述的:

ARC provides @autoreleasepool blocks instead. These have an advantage of being more efficient than NSAutoreleasePool. 

五、* 使用ARC需要遵守的新规则 *

①不要在dealloc方法中调用[super dealloc];

②不能使用 retain/release/retainCount/autorelease

③不能使用 NSAllocateObject/NSDeallocateObject

④不能使用 NSZone

⑤Objective-C 对象不能作为C语言结构体(struct/union)的成员

参考链接:

  1. arc & mrc这不是技术路线问题。
  2. Objective-c的内存管理MRC与ARC
  3. ARC使用心得(二)
  4. Performance Comparison: cocos2d-iphone v2 vs v3 vs Sparrow and ARC vs MRC
  5. iPhone开发之深入浅出 (4) — ARC之循环参照
  6. Transitioning to ARC Release Notes
时间: 2024-12-19 13:27:39

iOS arc VS mrc学习笔记的相关文章

《iOS应用逆向工程》学习笔记(四)iOS程序类型

越狱iOS中最常见的程序有Application, Dynamic Library和Daemon三类. 1.Application 除了传统意义上的App外,越狱iOS平台上还有两种App形式的存在:WeeApp(依附于NotificationCenter的App)和PreferenceBundle(依附于Settings的App),常见于Cydia平台. 普通App的bundle中存放的是可执行程序和所需资源,而framework的bundle中存放的是动态链接库. 主要关注App中的三个部分

《iOS应用逆向工程》学习笔记(五)初尝越狱插件OpenSSH

首先在越狱机子上装上OpenSSH插件,然后查看设备的IP地址,这里假设为192.168.xxx.xxx. 然后用Mac上的Terminal通过Open SSH连接到设备上(初次登录密码是alpine,必须立即修改,否则任何人都可以连接到你的机子上搞破坏). 连接命令为:ssh [email protected]设备IP地址 修改密码命令为:passwd 例如: $ ssh [email protected] The authenticity of host '192.168.xxx.xxx (

《iOS应用逆向工程》学习笔记(六)使用dumpdecrypted砸壳

本来是打算用AppCrackr砸壳的,结果砸壳都是失败的,开始以为是App的加密太厉害了,后来才知道是因为AppCrackr太暴力了,引起公愤,结果被人投诉招致核心功能被迫关闭了. 幸好在RE官网搜到一个用dumpdecrypted砸壳的帖子.下面是我砸壳的经历. 一.造锤 1.下载dumpdecrypted源码 下载地址:https://github.com/stefanesser/dumpdecrypted/archive/master.zip,接着在Mac中解压. 2.确认iOS设备的版本

《iOS应用逆向工程》学习笔记(二)iOS系统目录结构(部分)

首先下载个iFile,可以用来直观地查看iOS系统的目录结构. 下面记录一些关键的iOS目录结构: /var:"variable"的简写,存放一些经常更改的文件,例如日志.用户数据.临时文件等.其中/var/mobile/Applications下存放了所有App Store App. /Applications:存放所有的系统App和来自Cydia的App,不包括App Store App.越狱的过程把/Applications变成了一个符号链接,实际指向/var/stash/App

《iOS应用逆向工程》学习笔记(三)iOS文件权限

一个组可以包含多个用户,一个用户可以属于多个组. iOS中的每个文件都有一个属主用户和属主组.每个文件都具有一系列的权限. 在iOS中用3bit表示文件的使用权,从高位到低位分别是r(read), w(write), x(execute)权限. 文件和用户的关系存在三种可能性:(1)此用户是属主用户:(2)此用户不是属主用户,但在属主组里:(3)此用户既不是属主用户,也不在属主组里. 例如: 111101101代表rwxr-xr-x,其8进制为755,它表示该文件的属主用户拥有r, w, x权限

iOS ARC与MRC混编的一些解决方法

1. ARC & MRC 混合开发 在项目开发中,遇到使用MRC开发的第三方库怎么办? 例如:ASI 1> 尝试使用Xcode的转换工具(失败率比较高) 2> 在编译选项中,为MRC的程序添加-fno-objc-arc标记,表明在编译时,该文件使用MRC编译 备注: (1) 演示中使用的RegexKitLite还需要导入libicucore.dylib动态库 (2) 如果要在MRC项目中添加ARC的文件,可以使用 -fobjc-arc 标记即可 3> 将MRC的第三方库直接编译成

[iOS]ARC和MRC下混编

1.在MRC工程中使用ARC的文件(例如AFNetworking,SDWebImage,MJRefresh等)在Build Phases里找到对应.m 在后面添加-fobjc-arc(代表这个文件使用ARC进行编译) 2.在ARC工程中使用MRC文件,找到对应的文件在Build Phases中 找到对应的.m 在后面修改成-fno-objc-arc (这些.m使用MRC编译)

iOS APP开发概述----学习笔记001

之前开发过一些Android APP,现在开始学习iOS开发,未来实际工作应该会用到,未雨绸缪. 一.了解其系统层次架构 其系统分层四层,其详细如下: 二.开发平台组建 三.动手实践 可以自己动手,结合swift和MVC框架,写一个计算机的小例子. 版权声明:本文为博主原创文章,未经博主允许不得转载.

iOS ARC和MRC混编

如果一个工程为MRC,其中需要添加ARC的文件: 选择target  ->  build phases  ->  compile sources  ->单击ARC的文件将compile flags设置为: -fobjc - arc 如果一个工程为ARC,其中要添加MRC的文件: 选择target  ->  build phases  ->  compile sources  ->单击MRC的文件将compile flags设置为: - fno -objc -arc