内存管理策略

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

相关文章

Java内存泄漏问题的排查、优化与最佳实践

《Java内存泄漏问题的排查、优化与最佳实践》在Java开发中,内存泄漏是一个常见且令人头疼的问题,内存泄漏指的是程序在运行过程中,已经不再使用的对象没有被及时释放,从而导致内存占用不断增加,最终... 目录引言1. 什么是内存泄漏?常见的内存泄漏情况2. 如何排查 Java 中的内存泄漏?2.1 使用 J

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

如何测试计算机的内存是否存在问题? 判断电脑内存故障的多种方法

《如何测试计算机的内存是否存在问题?判断电脑内存故障的多种方法》内存是电脑中非常重要的组件之一,如果内存出现故障,可能会导致电脑出现各种问题,如蓝屏、死机、程序崩溃等,如何判断内存是否出现故障呢?下... 如果你的电脑是崩溃、冻结还是不稳定,那么它的内存可能有问题。要进行检查,你可以使用Windows 11

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分配。