概述

我们知道在程序运行过程中要创建大量的对象,和其他高级语言类似,在ObjC中对象时存储在堆中的,系统并不会自动释放堆中的内存(注意基本类型是由系统自己管理的,放在栈上)。如果一个对象创建并使用后没有得到及时释放那么就会占用大量内存。其他高级语言如C#、Java都是通过垃圾回收来(GC)解决这个问题的,但在OjbC中并没有类似的垃圾回收机制,因此它的内存管理就需要由开发人员手动维护。今天将着重介绍ObjC内存管理:

  1. 引用计数器
  2. 属性参数
  3. 自动释放池

引用计数器

在Xcode4.2及之后的版本中由于引入了ARC(Automatic Reference Counting)机制,程序编译时Xcode可以自动给你的代码添加内存释放代码,如果编写手动释放代码Xcode会报错,因此在今天的内容中如果你使用的是Xcode4.2之后的版本(相信现在大部分朋友用的版本都比这个要高),必须手动关闭ARC,这样才有助于你理解ObjC的内存回收机制。

ObjC中的内存管理机制跟C语言中指针的内容是同样重要的,要开发一个程序并不难,但是优秀的程序则更测重于内存管理,它们往往占用内存更少,运行更加流畅。虽然在新版Xcode引入了ARC,但是很多时候它并不能完全解决你的问题。在Xcode中关闭ARC:项目属性—Build Settings--搜索“garbage”找到Objective-C Automatic Reference Counting设置为No即可。

内存管理原理

我们都知道在C#、Java中都有GC在自动管理内存,当我们实例化一个对象之后通常会有一个变量来引用这个对象(变量中存储对象地址),当这个引用变量不再使用之后(也就是不再引用这个对象)此时GC就会自动回收这个对象,简单的说就是:当一个对象没有任何变量引用的时候就会被回收。

例如下面的C#代码片段

using System;namespace GC
{class Program{private static void Test(){object o=new object();}static void Main(string[] args){Test();}}
}

上面是一段C#代码,在Test()方法中,通过new Object()创建了一个对象,o是一个对象的引用(存储了对象的地址),它是一个局部变量,作用范围就是Test()方法内部。

当执行完Test()方法之后o就会被释放,此时由于没有变量在引用new Object()这个对象,因此GC会自动回收这个对象所占用的空间。

但是在ObjC中没有垃圾回收机制,那么ObjC中内存又是如何管理的呢?其实在ObjC中内存的管理是依赖对象引用计数器来进行的:在ObjC中每个对象内部都有一个与之对应的整数(retainCount),叫“引用计数器”,当一个对象在创建之后它的引用计数器为1,当调用这个对象的alloc、retain、new、copy方法之后引用计数器自动在原来的基础上加1(ObjC中调用一个对象的方法就是给这个对象发送一个消息),当调用这个对象的release方法之后它的引用计数器减1,如果一个对象的引用计数器为0,则系统会自动调用这个对象的dealloc方法来销毁这个对象。

下面通过一个简单的例子看一下引用计数器的知识:

Person.h

//
//  Person.h
//  MemoryManage
//
//  Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>@interface Person : NSObject#pragma mark - 属性
@property (nonatomic,copy) NSString *name;
@property (nonatomic,assign) int age;@end

Person.m

//
//  Person.m
//  MemoryManage
//
//  Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import "Person.h"@implementation Person#pragma mark - 覆盖方法
#pragma mark 重写dealloc方法,在这个方法中通常进行对象释放操作
-(void)dealloc{NSLog(@"Invoke Person's dealloc method.");[super dealloc];//注意最后一定要调用父类的dealloc方法(两个目的:一是父类可能有其他引用对象需要释放;二是:当前对象真正的释放操作是在super的dealloc中完成的)
}@end

main.m

//
//  main.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>
#import "Person.h"void Test1(){Person *p=[[Person alloc]init]; //调用alloc,引用计数器+1p.name=@"Kenshin";p.age=28;NSLog(@"retainCount=%lu",[p retainCount]);//结果:retainCount=1[p release];//结果:Invoke Person's dealloc method.//上面调用过release方法,p指向的对象就会被销毁,但是此时变量p中还存放着Person对象的地址,//如果不设置p=nil,则p就是一个野指针,它指向的内存已经不属于这个程序,因此是很危险的p=nil;//如果不设置p=nil,此时如果再调用对象release会报错,但是如果此时p已经是空指针了,//则在ObjC中给空指针发送消息是不会报错的[p release];
}void Test2(){Person *p=[[Person alloc]init];p.name=@"Kenshin";p.age=28;NSLog(@"retainCount=%lu",[p retainCount]);//结果:retainCount=1[p retain];//引用计数器+1NSLog(@"retainCount=%lu",[p retainCount]);//结果:retainCount=2[p release];//调用1次release引用计数器-1NSLog(@"retainCount=%lu",[p retainCount]);//结果:retainCount=1[p release];//结果:Invoke Person's dealloc method.p=nil;
}int main(int argc, const char * argv[]) {@autoreleasepool {Test1();}return 0;
}

在上面的代码中我们可以通过dealloc方法来查看是否一个对象已经被回收,如果没有被回收则有可能造成内存泄露。如果一个对象被释放之后,那么最后引用它的变量我们手动设置为nil,否则可能造成野指针错误,而且需要注意在ObjC中给空对象发送消息是不会引起错误的。

野指针错误形式在Xcode中通常表现为:Thread 1:EXC_BAD_ACCESS(code=EXC_I386_GPFLT)错误。因为你访问了一块已经不属于你的内存。

内存释放的原则

手动管理内存有时候并不容易,因为对象的引用有时候是错综复杂的,对象之间可能互相交叉引用,此时需要遵循一个法则:谁创建,谁释放

假设现在有一个人员Person类,每个Person可能会购买一辆汽车Car,通常情况下购买汽车这个活动我们可能会单独抽取到一个方法中,同时买车的过程中我们可能会多看几辆来最终确定理想的车,现在我们的代码如下:

Car.h

//
//  Car.h
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>@interface Car : NSObject#pragma mark - 属性
#pragma mark 车牌号
@property (nonatomic,copy) NSString *no;#pragma mark - 公共方法
#pragma mark 运行方法
-(void)run;@end

Car.m

//
//  Car.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import "Car.h"@implementation Car#pragma mark - 公共方法
#pragma mark 运行方法
-(void)run{NSLog(@"Car(%@) run.",self.no);
}#pragma mark - 覆盖方法
#pragma mark 重写dealloc方法
-(void)dealloc{NSLog(@"Invoke Car(%@) dealloc method.",self.no);[super dealloc];
}
@end

Person.h

//
//  Person.h
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>
@class Car;@interface Person : NSObject{Car *_car;
}#pragma mark - 属性
#pragma mark 姓名
@property (nonatomic,copy) NSString *name;#pragma mark - 公共方法
#pragma mark Car属性的set方法
-(void)setCar:(Car *)car;
#pragma mark  Car属性的get方法
-(Car *)car;
@end

Person.m

//
//  Person.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import "Person.h"
#import "Car.h"@implementation Person#pragma mark - 公共方法
#pragma mark Car属性的set方法
-(void)setCar:(Car *)car{if (_car!=car) { //首先判断要赋值的变量和当前成员变量是不是同一个变量[_car release]; //释放之前的对象_car=[car retain];//赋值时重新retain}
}
#pragma mark  Car属性的get方法
-(Car *)car{return _car;
}#pragma mark - 覆盖方法
#pragma mark 重写dealloc方法
-(void)dealloc{NSLog(@"Invoke Person(%@) dealloc method.",self.name);[_car release];//在此释放对象,即使没有赋值过由于空指针也不会出错[super dealloc];
}
@end

main.m

//
//  main.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>
#import "Person.h"
#import "Car.h"void getCar(Person *p){Car *car1=[[Car alloc]init];car1.no=@"888888";p.car=car1;NSLog(@"retainCount(p)=%lu",[p retainCount]);Car *car2=[[Car alloc]init];car2.no=@"666666";[car1 release];car1=nil;[car2 release];car2=nil;
}int main(int argc, const char * argv[]) {@autoreleasepool {Person *p=[[Person alloc]init];p.name=@"Kenshin";getCar(p);[p.car run];[p release];p=nil;}return 0;
}

程序运行结果:

从运行结果来看创建的三个对象p、car1、car2都被回收了,而且[p.car run]也能顺利运行,已经达到了我们的需求。但是这里需要重点解释一下setCar方法的实现,setCar方法中为什么没有写成如下形式:

-(void)setCar:(Car *)car{_car=car;
}

前面在我们说到属性的定义时不是都采用的这种方式吗?

根据前面说到的内存释放原则,getCar方法完全符合,在这个方法中定义的两个对象car1、car2也都是在这个方法中释放的,包括main函数中的p对象也是在main函数中定义和释放的。但是如果发现调用完getCar方法之后紧接着调用了汽车的run方法,当然这在程序设计和开发过程中应该是再普通不过的设计了。如果setCar写成“_car=car”的形式,当调用完getCar方法后,人员的car属性被释放了,此时调用run方法是会报错的(大家自己可以试试)。但是如下的方式却不会有问题:

-(void)setCar:(Car *)car{if (_car!=car) { //首先判断要赋值的变量和当前成员变量是不是同一个变量[_car release]; //释放之前的对象_car=[car retain];//赋值时重新retain}
}

因为在这个方法中我们通过[car retain]保证每次属性赋值的时候对象引用计数器+1,这样一来调用过getCar方法可以保证人员的car属性不会被释放,其次为了保证上一次的赋值对象(car1)能够正常释放,我们在赋新值之前对原有的值进行release操作。最后在Person的dealloc方法中对_car进行一次release操作(因为setCar中做了一次retain操作)保证_car能正常回收。

属性参数

像上面这样编写setCar方法的情况是比较多的,那么如何使用@property进行自动实现呢?答案就是使用属性参数,例如上面car属性的setter方法,可以通过@property定义如下:

@property (nonatomic,retain) Car *car;

你会发现此刻我们不必手动实现car的getter、setter方法程序仍然没有内存泄露。其实大家也应该都已经看到前面Person的name属性定义的时候我们同样加上了(nonatomic,copy)参数,这些参数到底是什么意思呢?

@property的参数分为三类,也就是说参数最多可以有三个,中间用逗号分隔,每类参数可以从上表三类参数中人选一个。如果不进行设置或者只设置其中一类参数,程序会使用三类中的各个默认参数,默认参数:(atomic,readwrite,assign)

一般情况下如果在多线程开发中一个属性可能会被两个及两个以上的线程同时访问,此时可以考虑atomic属性,否则建议使用nonatomic,不加锁,效率较高;readwirte方法会生成getter、setter两个方法,如果使用readonly则只生成getter方法;关于set方法处理需要特别说明,假设我们定义一个属性a,这里列出三种方式的生成代码:

assign,用于基本数据类型

-(void)setA:(int)a{_a=a;
}

retain,通常用于非字符串对象

-(void)setA:(Car *)a{if(_a!=a){[_a release];_a=[a retain];}
}

copy,通常用于字符串对象

-(void)setA:(NSString *)a{if(_a!=a){[_a release];_a=[a copy];}
}

自动释放池

在ObjC中也有一种内存自动释放的机制叫做“自动引用计数”(或“自动释放池”),与C#、Java不同的是,这只是一种半自动的机制,有些操作还是需要我们手动设置的。自动内存释放使用@autoreleasepool关键字声明一个代码块,如果一个对象在初始化时调用了autorelase方法,那么当代码块执行完之后,在块中调用过autorelease方法的对象都会自动调用一次release方法。这样一来就起到了自动释放的作用,同时对象的销毁过程也得到了延迟(统一调用release方法)。看下面的代码:

Person.h

//
//  Person.h
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>@interface Person : NSObject#pragma mark - 属性
#pragma mark 姓名
@property (nonatomic,copy) NSString *name;#pragma mark - 公共方法
#pragma mark 带参数的构造函数
-(Person *)initWithName:(NSString *)name;
#pragma mark 取得一个对象(静态方法)
+(Person *)personWithName:(NSString *)name;
@end

Person.m

//
//  Person.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import "Person.h"@implementation Person#pragma mark - 公共方法
#pragma mark 带参数的构造函数
-(Person *)initWithName:(NSString *)name{if(self=[super init]){self.name=name;}return self;
}
#pragma mark 取得一个对象(静态方法)
+(Person *)personWithName:(NSString *)name{Person *p=[[[Person alloc]initWithName:name] autorelease];//注意这里调用了autoreleasereturn p;
}#pragma mark - 覆盖方法
#pragma mark 重写dealloc方法
-(void)dealloc{NSLog(@"Invoke Person(%@) dealloc method.",self.name);[super dealloc];
}@end

main.m

//
//  main.m
//  MemoryManage
//
//  Created by Kenshin Cui on 14-2-15.
//  Copyright (c) 2014年 Kenshin Cui. All rights reserved.
//#import <Foundation/Foundation.h>
#import "Person.h"int main(int argc, const char * argv[]) {@autoreleasepool {Person *person1=[[Person alloc]init];[person1 autorelease];//调用了autorelease方法后面就不需要手动调用release方法了person1.name=@"Kenshin";//由于autorelease是延迟释放,所以这里仍然可以使用person1Person *person2=[[[Person alloc]initWithName:@"Kaoru"] autorelease];//调用了autorelease方法Person *person3=[Person personWithName:@"rosa"];//内部已经调用了autorelease,所以不需要手动释放,这也符合内存管理原则,因为这里并没有alloc所以不需要release或者autoreleasePerson *person4=[Person personWithName:@"jack"];[person4 retain];}/*结果:Invoke Person(rosa) dealloc method.Invoke Person(Kaoru) dealloc method.Invoke Person(Kenshin) dealloc method.*/return 0;
}

当上面@autoreleaespool代码块执行完之后,三个对象都得到了释放,但是person4并没有释放,原因很简单,由于我们手动retain了一次,当自动释放池释放后调用四个对的release方法,当调用完person4的release之后它的引用计数器为1,所有它并没有释放(这是一个反例,会造成内存泄露);autorelase方法将一个对象的内存释放延迟到了自动释放池销毁的时候,因此上面person1,调用完autorelase之后它还存在,因此给name赋值不会有任何问题;在ObjC中通常如果一个静态方法返回一个对象本身的话,在静态方法中我们需要调用autorelease方法,因为按照内存释放原则,在外部使用时不会进行alloc操作也就不需要再调用release或者autorelase,所以这个操作需要放到静态方法内部完成。

对于自动内存释放简单总结一下:

  1. autorelease方法不会改变对象的引用计数器,只是将这个对象放到自动释放池中;
  2. 自动释放池实质是当自动释放池销毁后调用对象的release方法,不一定就能销毁对象(例如如果一个对象的引用计数器>1则此时就无法销毁);
  3. 由于自动释放池最后统一销毁对象,因此如果一个操作比较占用内存(对象比较多或者对象占用资源比较多),最好不要放到自动释放池或者考虑放到多个自动释放池;
  4. ObjC中类库中的静态方法一般都不需要手动释放,内部已经调用了autorelease方法;
本作品采用知识共享署名 2.5 中国大陆许可协议进行许可,欢迎转载,演绎或用于商业目的。但转载请注明来自崔江涛(KenshinCui),并包含相关链接。
绿色通道: 好文要顶 关注我 收藏该文与我联系 

KenshinCui
关注 - 3
粉丝 - 1643

荣誉:推荐博客
+加关注

14
0
(请您对文章做出评价)

« 上一篇:IOS开发系列--Objective-C之协议、代码块、分类
» 下一篇:IOS开发系列--Objective-C之KVC、KVO

  • 分类: IOS, Objective-C
  • 标签: copy, 内存管理, alloc, retain, new, release, autorelease, autoreleasepool
ADD YOUR COMMENT

  1. #1楼 @@yuan  2014-08-13 14:03
  2. #2楼[楼主] KenshinCui  2014-08-14 23:06
  3. #3楼 lichuanghan  2014-10-18 13:19
  4. #4楼[楼主] KenshinCui  2015-01-21 16:10
  5. #5楼 mefive  2015-01-22 19:20
    @lichuanghan

    若是在方法内的变量,其他地方并无机会使用。确定不用后,release。便无需再设置为nil了。

  6. #6楼 luckyqlxx  2015-02-09 17:56
  7. #7楼 深谷芝兰  2015-05-15 14:22
  8. #8楼 Weiq.Lv  2015-06-07 18:55
  9. #9楼 yltan_net  2015-06-08 11:19
  10. #10楼[楼主] KenshinCui  2015-06-17 10:04
  11. #11楼[楼主] KenshinCui  2015-06-17 10:05
  12. #12楼[楼主] KenshinCui  2015-06-17 10:05
  13. #13楼 daisy1227  2015-07-03 16:36
  14. #14楼 daisy1227  2015-07-03 16:40
  15. #15楼 金桔檸檬  2015-07-07 16:01
    受益匪浅,博主不使用@property设置Car对象,详细设置setter和getter方法,深入本质给大家讲解,很精辟,用心良苦。
    博主,有个问题请教一下:字符串对象需要释放吗?
    如果@property (nonatimic, copy) NSString *name;
    相当于: - (voie)setName:(NSString *)name {
    if (_name != name) {
    [_name release];
    _name = [name copy];
    }
    }
    void Test1(){
    Person *p=[[Person alloc]init]; //调用alloc,引用计数器+1
    p.name=@"Kenshin"; //调用copy,浅拷贝,@“kenshin”的retainCount就是多少,因为没有用到alloc创建然后赋值,这么写学生不太懂
    如果是:Person *p=[[Person alloc]init]; //调用alloc,引用计数器+1
    NSString *s = [[NSString alloc] initWithString:@"Kenshin"];
    p.name = s; // @"Kenshin" retainCount为2
    如果是:Person *p=[[Person alloc]init]; //调用alloc,引用计数器+1
    NSString *s = [NSString stringWithString:@"Kenshin"];
    //或者NSString *s = @"Kenshin";都没用到alloc
    p.name = s; // @"Kenshin" retainCount为????

IOS开发系列—Objective-C之内存管理相关推荐

  1. iOS 与OS X多线程和内存管理 笔记 ARC与所有权修饰符

    注:本文为笔记形式,所以很多都是摘抄的.<<iOS 与OS X多线程和内存管理>>书中写的很棒,简单易懂,建议各位看官自己去看看. ####ARC和MRC 前一篇主要是MRC环 ...

  2. iOS开发系列--UITableView全面解析

    iOS开发系列--UITableView全面解析 2014-08-23 23:20 by KenshinCui, 2202 阅读, 18 评论, 收藏,  编辑 --UIKit之UITableView ...

  3. IOS开发系列--IOS程序开发概览

    IOS开发系列--IOS程序开发概览 2014-08-04 19:42 by KenshinCui, 9983 阅读, 51 评论, 收藏, 编辑 概览 终于到了真正接触IOS应用程序的时刻了,之前我 ...

  4. iOS开发系列--网络开发(转)

    iOS开发系列--网络开发 2014-10-22 08:34 by KenshinCui, 66365 阅读, 56 评论, 收藏, 编辑 概览 大部分应用程序都或多或少会牵扯到网络开发,例如说新浪微 ...

  5. iOS开发系列文章(持续更新……)

    iOS开发系列的文章,内容循序渐进,包含C语言.ObjC.iOS开发以及日后要写的游戏开发和Swift编程几部分内容.文章会持续更新,希望大家多多关注,如果文章对你有帮助请点赞支持,多谢! 为了方便大 ...

  6. iOS开发系列--数据存取

    原文地址为: iOS开发系列--数据存取 概览 在iOS开发中数据存储的方式可以归纳为两类:一类是存储为文件,另一类是存储到数据库.例如前面IOS开发系列-Objective-C之Foundation ...

  7. iOS开发系列--通讯录、蓝牙、内购、GameCenter、iCloud、Passbook详解

    代码改变世界 Posts - 69, Articles - 0, Comments - 812 Cnblogs Dashboard Login Home Contact Gallery RSS Ken ...

  8. iOS开发系列--IOS程序开发概览ios基础

    iOS开发系列--IOS程序开发概览 概览 终于到了真正接触IOS应用程序的时刻了,之前我们花了很多时间去讨论C语言.ObjC等知识,对于很多朋友而言开发IOS第一天就想直接看到成果,看到可以运行的I ...

  9. iOS开发系列--网络开发

    概览 大部分应用程序都或多或少会牵扯到网络开发,例如说新浪微博.微信等,这些应用本身可能采用iOS开发,但是所有的数据支撑都是基于后台网络服务器的.如今,网络编程越来越普遍,孤立的应用通常是没有生命力 ...

最新文章

  1. (C++)201709-1 打酱油
  2. JSP -- JSP语法
  3. DataTable实现分组
  4. java怎么连elk_从Java应用程序登录到ELK而无需解析日志
  5. pandas插入新列
  6. Linux下部署LVS(DR)+keepalived+Nginx负载均衡
  7. 关于动态语言 静态语言 静态类型语言 动态类型语言的区别
  8. Python3 列表
  9. DELL服务器装机网络问题
  10. Mysql常用命令笔记
  11. 教你查看传说中的WPS2005彩蛋
  12. 数字孪生智慧城市信息化平台建设
  13. meta的http-equiv属性
  14. 《简单的逻辑学》阅读笔记(思维导图)
  15. 小学五年级如何引导学计算机,怎么教五年级小孩数学
  16. 最简单优雅修改jupyter自带主题字体大小颜色(不需要插件不需要第三方theme)
  17. Cross-X Learning for Fine-Grained Visual Categorization
  18. 用tensorflow深度学习梵高的画并模仿
  19. Vue3使用富文本框(wangeditor)
  20. 离职通知邮件主题写什么好_“辞职”发邮件标题怎么写?

热门文章

  1. java程序当当网购书系统怎么运行_Java实现基于控制台的购书系统基本操作
  2. 回溯法求解N皇后问题
  3. 在线预览文档html版
  4. 应用技术大公开系列Q之十一:(纤维).石墨烯纸制备工艺 (*3-4)
  5. vue项目点击左侧子菜单,打开一个新的浏览器标签页
  6. UI设计师是吃青春饭的? 我来让你彻底了解了解UI!
  7. opencv-python最全下载地址
  8. Flink的背压机制
  9. 【已解决】计算机丢失d3d12.dll怎么办?d3d12.dll加载失败怎么办
  10. java 监听器作用_浅谈java监听器的作用