内存管理策略

2024-04-02 11:48
文章标签 内存 管理策略

本文主要是介绍内存管理策略,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

内存管理策略

在引用计数的环境下管理内存使用的基本模型是,通过在NSObject协议定义的方法和提供标准命名的方法。NSObject类也定义了一个方法“dealloc”,当一个对象被释放时此函数被调用。本文介绍了您需要知道的,如何正确的管理内存在一个Cocoa程序,并提供了一些正确的使用实例。

基本内存管理规则

内存管理模型是基于对象所有权的。任何一个对象可能会有一个或者多个所有者。只要一个对象有至少一个所有者,那么它需要继续退出。当一个对象没有所有者时,系统会自动销毁它。

为了确保你是否清除一个对象,Cocoa提供以下策略:

拥有任何一个创建的对象

用“alloc”,“new”,“copy”,或“mutablecopy”方法创建对象。(例如,alloc,newobject,或mutablecopy)。

可以使用retain方法保留对象的所有权

通过retain这个方法可以有效地保证原对象被引用,这个方法也安全地放回原对象的调用者。可以在以下两个情况下使用retain方法:
(1)实现一个访问对象的方法或者init方法,为了得到想保存对象所有权作为属性时。
(2)为了防止一些其他操作造成对象被废弃。(如:使用了一个被废弃的对象)

当不再需要它时,必须放弃对象的所有权

通过发送release消息或autorelease消息放弃对象的所有权。在Cocoa术语中,放弃对象被称为“释放”对象。

你不能释放你没有所有权的对象

这是前面规则的推论。

一个简单的例子:

为了说明该策略,考虑下面的代码片段:

{Person *aPerson = [[Person alloc] init];// ...NSString *name = aPerson.fullName;// ...[aPerson release];
}

这个Person对象是由alloc方法创建的,所以随后当不需要时发送release消息进行释放。这个人的名字变量没有被任何一个方法所取用,所以它不用发送release消息。注意,这个例子使用的是释放而不是自动释放。

用autorelease发送一个延迟释放

当需要放送一个延迟释放消息时,你可以使用autorelease。这通常是在返回一个对象方法时使用。下面举一个例子,你可以实现获取人的名字方法像这样:

- (NSString *)fullName {NSString *string = [[[NSString alloc] initWithFormat:@"%@ %@",self.firstName, self.lastName] autorelease];return string;
}

通过alloc方法获取到了这个string。要遵守内存管理策略,那么当不再引用对象时,必须放弃所有权。如果使用release,那么此string将要在返回前被析构掉了(此时该函数将返回一个无效的对象)。使用autorelease就意味着想要释放所有权,但允许调用方法在该对象释放前使用。

也可以用下面的方式实现:

- (NSString *)fullName {NSString *string = [NSString stringWithFormat:@"%@ %@",self.firstName, self.lastName];return string;
}

根据基本的规则,由于没有从stringWithFormat方法得到返回string的所有权,所以可以放心地通过此方法返回string。

相比之下,下面的实现方式是错误的:

- (NSString *)fullName {NSString *string = [[NSString alloc] initWithFormat:@"%@ %@",self.firstName, self.lastName];return string;
}

根据命令规定,没有说明fullName方法的调用方拥有返回字符串的所有权。因此调用方没有理由释放字符串,这样就造成内存泄漏。

不能拥有引用对象的所有权

在Cocoa框架中有一些特殊的方法,他们返回对象的引用(他们的参数类型可能是ClassName**或者id*)。比较常见的方法就是,当发生了一个错误,通过NSError保存出错信息。如下所示:

initWithContentsOfURL:options:error: (NSData) and initWithContentsOfFile:encoding:error: (NSString).

在这些情况下,同样的规则也适用于此。当调用这些方法,你没有创建对象,那么你没有对象的所有权,所以也不需要释放它。如下例子所示:

NSString *fileName = <#Get a file name#>;
NSError *error;
NSString *string = [[NSString alloc] initWithContentsOfFile:fileNameencoding:NSUTF8StringEncoding error:&error];
if (string == nil) {// Deal with error...
}
// ...
[string release];

实现dealloc方法去释放对象所有权

NSObject类定义了dealloc方法。当对象没有拥有者,或者内存被回收时,该方法自动调用。这个方法的作用是释放对象的内存,并处理它拥有的任何资源,包括任何对象实例变量的所有权。

下面这个例子描述怎么实现Person类的dealloc方法:

@interface Person : NSObject
@property (retain) NSString *firstName;
@property (retain) NSString *lastName;
@property (assign, readonly) NSString *fullName;
@end@implementation Person
// ...
- (void)dealloc[_firstName release];[_lastName release];[super dealloc];
}
@end

注意:不要直接调用任何对象的dealloc方法。

在dealloc函数实现方法结束前,要调用父类的dealloc方法。

不应该把系统资源的管理放到对象生命周期中;不用用dealloc方法管理稀缺资源。

当应用程序中断时,对象不可能发送dealloc消息。因为进程的内存在退出时自动清除,所以更高效的操作系统来清除资源,而不是调用所有的内存管理方法。

核心库使用类似但不同的规则

有类似的管理规则给核心库对象。Cocoa和核心库的命名规则是不同的。特别是核心库的创建规则不适用于返回Objective-C对象的方法。

举一个例子,在下面的代码段中,你不负责释放myInstance实例所有权的。

MyClass *myInstance = [MyClass createInstance];




这篇关于内存管理策略的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/869897

相关文章

NameNode内存生产配置

Hadoop2.x 系列,配置 NameNode 内存 NameNode 内存默认 2000m ,如果服务器内存 4G , NameNode 内存可以配置 3g 。在 hadoop-env.sh 文件中配置如下。 HADOOP_NAMENODE_OPTS=-Xmx3072m Hadoop3.x 系列,配置 Nam

JVM内存调优原则及几种JVM内存调优方法

JVM内存调优原则及几种JVM内存调优方法 1、堆大小设置。 2、回收器选择。   1、在对JVM内存调优的时候不能只看操作系统级别Java进程所占用的内存,这个数值不能准确的反应堆内存的真实占用情况,因为GC过后这个值是不会变化的,因此内存调优的时候要更多地使用JDK提供的内存查看工具,比如JConsole和Java VisualVM。   2、对JVM内存的系统级的调优主要的目的是减少

JVM 常见异常及内存诊断

栈内存溢出 栈内存大小设置:-Xss size 默认除了window以外的所有操作系统默认情况大小为 1MB,window 的默认大小依赖于虚拟机内存。 栈帧过多导致栈内存溢出 下述示例代码,由于递归深度没有限制且没有设置出口,每次方法的调用都会产生一个栈帧导致了创建的栈帧过多,而导致内存溢出(StackOverflowError)。 示例代码: 运行结果: 栈帧过大导致栈内存

理解java虚拟机内存收集

学习《深入理解Java虚拟机》时个人的理解笔记 1、为什么要去了解垃圾收集和内存回收技术? 当需要排查各种内存溢出、内存泄漏问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就必须对这些“自动化”的技术实施必要的监控和调节。 2、“哲学三问”内存收集 what?when?how? 那些内存需要回收?什么时候回收?如何回收? 这是一个整体的问题,确定了什么状态的内存可以

NGINX轻松管理10万长连接 --- 基于2GB内存的CentOS 6.5 x86-64

转自:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=190176&id=4234854 一 前言 当管理大量连接时,特别是只有少量活跃连接,NGINX有比较好的CPU和RAM利用率,如今是多终端保持在线的时代,更能让NGINX发挥这个优点。本文做一个简单测试,NGINX在一个普通PC虚拟机上维护100k的HTTP

PHP原理之内存管理中难懂的几个点

PHP的内存管理, 分为俩大部分, 第一部分是PHP自身的内存管理, 这部分主要的内容就是引用计数, 写时复制, 等等面向应用的层面的管理. 而第二部分就是今天我要介绍的, zend_alloc中描写的关于PHP自身的内存管理, 包括它是如何管理可用内存, 如何分配内存等. 另外, 为什么要写这个呢, 因为之前并没有任何资料来介绍PHP内存管理中使用的策略, 数据结构, 或者算法. 而在我们

string字符会调用new分配堆内存吗

gcc的string默认大小是32个字节,字符串小于等于15直接保存在栈上,超过之后才会使用new分配。

PHP内存泄漏问题解析

内存泄漏 内存泄漏指的是在程序运行过程中申请了内存,但是在使用完成后没有及时释放的现象, 对于普通运行时间较短的程序来说可能问题不会那么明显,但是对于长时间运行的程序, 比如Web服务器,后台进程等就比较明显了,随着系统运行占用的内存会持续上升, 可能会因为占用内存过高而崩溃,或被系统杀掉 PHP的内存泄漏 PHP属于高级语言,语言级别并没有内存的概念,在使用过程中完全不需要主动申请或释放内

C++学习笔记----6、内存管理(四)---- 通常的内存陷阱(2)

3、Windows环境下使用Visual C++发现并修复内存渗露         内存渗露很难跟踪是因为你无法很容易地看着内存并且看到什么对象处于使用中,一开始在哪儿分配的内存。然而,是有程序可以为你做到这一点的。内存渗露检测工具有昂贵的专业软件包,也有免费下载的工具。如果你是在Microsoft Visual C++环境下工作,它的排错工具库有内建的对于内存渗露检测的支持。该内存检测默认没有

控制台和MFC中内存泄露工具vld的使用

最近想检测下项目中内存泄露的情况,选中了vld这款。在查找使用方法的时候,大都是控制台下的示例,添加到main函数所在的源文件上。换成MFC就纠结了,不知道添加到哪里去。本文记录控制台和MFC中的使用vld过程。    vld资源:    1)、大家可以移步下边的网址下载:     http://vld.codeplex.com/releases/view/82311    2