内存管理策略

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

相关文章

Python如何使用__slots__实现节省内存和性能优化

《Python如何使用__slots__实现节省内存和性能优化》你有想过,一个小小的__slots__能让你的Python类内存消耗直接减半吗,没错,今天咱们要聊的就是这个让人眼前一亮的技巧,感兴趣的... 目录背景:内存吃得满满的类__slots__:你的内存管理小助手举个大概的例子:看看效果如何?1.

Redis 内存淘汰策略深度解析(最新推荐)

《Redis内存淘汰策略深度解析(最新推荐)》本文详细探讨了Redis的内存淘汰策略、实现原理、适用场景及最佳实践,介绍了八种内存淘汰策略,包括noeviction、LRU、LFU、TTL、Rand... 目录一、 内存淘汰策略概述二、内存淘汰策略详解2.1 ​noeviction(不淘汰)​2.2 ​LR

Golang基于内存的键值存储缓存库go-cache

《Golang基于内存的键值存储缓存库go-cache》go-cache是一个内存中的key:valuestore/cache库,适用于单机应用程序,本文主要介绍了Golang基于内存的键值存储缓存库... 目录文档安装方法示例1示例2使用注意点优点缺点go-cache 和 Redis 缓存对比1)功能特性

Go使用pprof进行CPU,内存和阻塞情况分析

《Go使用pprof进行CPU,内存和阻塞情况分析》Go语言提供了强大的pprof工具,用于分析CPU、内存、Goroutine阻塞等性能问题,帮助开发者优化程序,提高运行效率,下面我们就来深入了解下... 目录1. pprof 介绍2. 快速上手:启用 pprof3. CPU Profiling:分析 C

golang内存对齐的项目实践

《golang内存对齐的项目实践》本文主要介绍了golang内存对齐的项目实践,内存对齐不仅有助于提高内存访问效率,还确保了与硬件接口的兼容性,是Go语言编程中不可忽视的重要优化手段,下面就来介绍一下... 目录一、结构体中的字段顺序与内存对齐二、内存对齐的原理与规则三、调整结构体字段顺序优化内存对齐四、内

Linux内存泄露的原因排查和解决方案(内存管理方法)

《Linux内存泄露的原因排查和解决方案(内存管理方法)》文章主要介绍了运维团队在Linux处理LB服务内存暴涨、内存报警问题的过程,从发现问题、排查原因到制定解决方案,并从中学习了Linux内存管理... 目录一、问题二、排查过程三、解决方案四、内存管理方法1)linux内存寻址2)Linux分页机制3)

Java循环创建对象内存溢出的解决方法

《Java循环创建对象内存溢出的解决方法》在Java中,如果在循环中不当地创建大量对象而不及时释放内存,很容易导致内存溢出(OutOfMemoryError),所以本文给大家介绍了Java循环创建对象... 目录问题1. 解决方案2. 示例代码2.1 原始版本(可能导致内存溢出)2.2 修改后的版本问题在

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Redis多种内存淘汰策略及配置技巧分享

《Redis多种内存淘汰策略及配置技巧分享》本文介绍了Redis内存满时的淘汰机制,包括内存淘汰机制的概念,Redis提供的8种淘汰策略(如noeviction、volatile-lru等)及其适用场... 目录前言一、什么是 Redis 的内存淘汰机制?二、Redis 内存淘汰策略1. pythonnoe

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

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