Objective-C 【内存管理&手动内存管理 综述】

———————————————————————————————————————————

内存管理

(1)Objective-C的内存管理

栈区    存放局部变量(由于基本数据类型占用的存储空间是固定的,由系统去分配,我们不用去管,故栈区存放基本数据类型,)

堆区    存放程序运行过程中动态分配的内存空间(对象类型是程序运行过程中动态分配的,他们的大小不固定。比如说是我们Person new申请来的,存放在堆区,也是我们需要管理的)

★所以内存管理的范围是   继承了NSObject的所有对象(堆区中的对象)

★这是为什么呢?因为代码执行完毕后,堆内存 不会自动释放(需要内存管理机制),而 栈内存 会自动释放(自动弹栈)

引用计数器     (一般用%ld或者%tu去查看里面的值,无符号的)

每个对象都有自己的引用计数器,它是一个整数表示对象被引用的次数,即现在有多少东西在使用这个对象,引用计数器的值变为0时,对象就销毁

①每一个对象刚生下来的时候,引用计数器都是1(对象一旦创建好,默认的引用计数器就是1)

②如果对象的引用计数器不为0,那么在整个程序的运行过程中,它占用的内存就不可能被回收(除非整个程序退出)

③当使用alloc、new或者copy创建一个对象时,对象的引用计数器默认就是1

(2)Objective-C的内存管理分类

Objective-C提供了三种内存管理的方式:

①Mannul Reference Counting(MRC,手动管理,在开发 iOS 4.1之前的版本项目时我们要自己负责使用引用计数来管理内存,比如手动retain、release、autorelease等。而在iOS之后的版本我们可以使用ARC,让系统自己管理内存)

②Automatic Reference Counting(ARC,自动引用计数,iOS 4.1后推出)

③Garbage Collection(垃圾回收,iOS不支持垃圾回收)

★苹果官方推荐我们使用ARC技术来管理内存,MRC理解就好

———————————————————————————————————————————

手动内存管理(MRC)

(1)首先我们要手动内存管理(MRC),就要先将自动管理关闭(关闭ARC)。

关闭ARC的方法:

点击项目(总的,蓝色)——> bulid setting ——> basic level ——>搜索 Objective-C Automatic Reference Counting ——> 此时会出现四个选项,全部设为 No 即可

(2)我们判断对象是否需要回收,这是就要查看引用计数器的值,需要调用

-(NSInteger)retainCount方法(这是一个有返回值的对象方法,返回值为NSInteger,NSInteger是一个动态类型,他的类型在不同的操作系统可能是int类型,也可能是long类型,他是有符号的。而NSUInerger是无符号类型,没有负数)

retainCount的返回值  > 0   不会回收

retainCount的返回值  = 0   回收内存空间

★注意:这里retainCount的返回值是 %tu 或者是 %ld 类型的~

(3)回收对象内存空间的时候通常会自动调用一个方法,dealloc(释放对象的属性)

我们可以通过重写dealloc这个方法来看看到底对象空间有没有被释放(再里面加一句话输出,如果释放了就会自动调用dealloc,那么这句话就会输出)

★这里注意重写dealloc和重写init一样,要先调用父类的dealloc方法,即  [super dealloc]; ,原因不再多说

(4)有两个对象方法可以使对象的引用计数器的值增加1和减少1

①使用 release  可以使引用计数  -1

②使用 retain     可以使引用计数  +1

(5)内存管理

我们应该心里有这么一个观念,就是让对象的引用计数保持一个平衡的状态

即:    retain + new = release  (增加引用计数=减少引用计数)

代码:

#import <Foundation/Foundation.h>

@interface Person : NSObject

-(void)run;

@end

#import "Person.h"

@implementation Person

- (void)dealloc

{

NSLog(@"对象内存空间释放!");

[super dealloc];//这句话一定要放在重写dealloc方法的最后一句,意义是:先释放子类占用的内存空间,再释放父类占用的内存空间(和重写构造方法正好相反,初始化对象时是先执行父类的init,再执行子类的初始化)

//    另外注意,永远不要直接调用dealloc方法,如果调用应该让retainCount = 0,但是如果你写 [p dealloc] 也是不会报错的,只是没有必要!这压根就是一个不需要你自己闲的蛋疼去写的一句话。

}

-(void)run

{

NSLog(@"run!");

}

@end

int main(int argc, const char * argv[]) {

@autoreleasepool {

Person *p=[Person new];

NSLog(@"p->retainCount = %tu",[p retainCount]);

//        这个地方因为创建了p,所以引用p的次数+1,所以输出的值为1。但是记住,这里必须将ARC关闭,如果不关闭是会报错的

[p run];

NSLog(@"p->retainCount = %tu",[p retainCount]);

//        这里输出结果还是1,因为 [p run]; 只是调用了run方法,但是p被引用的次数还是1

[p release];

//        这里将引用计数-1,所以说此时引用计数为0,也就是此时将会自动调用dealloc方法,打印我们重写dealloc里面的“对象空间释放!”

//        [p run];   //对象空间都释放了,但还是可以调用的,这是为什么呢?这个问题留在下节“单个对象的内存管理(野指针)去解释”

}

return 0;

}

———————————————————————————————————————————

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-10-12 04:53:20

Objective-C 【内存管理&手动内存管理 综述】的相关文章

Objective-C中的内存管理——手动内存管理

一.引用计数(retainCount) 1.内存的申请 alloc和new是Objective-C语言中申请内存的两种常用方式. 当我们向一个类对象发送一个alloc消息的时候,其会向操作系统在堆中申请一块内存用于存放该类的实例对象.然后将该实例对象的引用计数(retainCount)设置为1,最后返回该对象的指针. alloc和new的区别:一句话,[Person new]即[[Person alloc]new] 2.引用计数(retainCount) Cocoa框架中,所有继承自NSObje

object-C 手动内存管理(MRC)

object-C的内存管理和javascript的垃圾回收不一样,今天总结下手动内存管理,ARC的后边补上. 1:基本铺垫 oc采用引用计数来表示对象的状态,比如通过init创建出来的一个对象引用计数为1,如果想让它释放则对这个对象发送一条release消息,则引用计数-1,那怎么+1呢,给这个对象发送retain消息. 对象操作 Object-C方法 生成并持有对象 alloc/new/copy/mutableCopy方法 持有对象 retain方法 释放对象 release方法 废弃对象 d

手动内存管理和自动释放池

手动内存管理 在进行内存管理的时候要注意内存管理的准则:谁开辟内存,谁释放内存(谁污染的谁治理) .开辟内存之后,对象的引用计数为1,只有继承自NSObject的对象才有内促管理的概念, 当对象引用计数为0的时候对象的内存会被清理. 下列关键字会开辟内存,对象引用计数器+1 alloc new copy mutableCopy 下列是内存管理的相关方法. retain :保留对象,对象的引用计数器+1. release : 释放对象,对象引用计数器-1. retainCount : 获取对象当前

iOS-旧项目中手动内存管理(MRC)转ARC

在ARC之前,iOS内存管理无论对资深级还是菜鸟级开发者来说都是一件很头疼的事.我参 加过几个使用手动内存管理的项目,印象最深刻的是一个地图类应用,由于应用本身就非常耗内存,当时为了解决内存泄露问题,每周都安排有人值班用 Instruments挨个跑功能,关键是每次都总能检查出来不少.其实不管是菜鸟级还是资深级开发者都避免不了写出内存泄露的代码,规则大家都懂,可是 天知道什么时候手一抖就少写了个release? 好在项目决定转成ARC了,下面将自己转换的过程和中间遇到的问题写出来和大家共享,希望

黑马程序员---OC基础6【内存管理】【手动内存管理】【单、多个对象的内存管理】【*@property参数】【@class的使用】【NSString类的内存管理】【autorelease使用】

------- iOS培训.Android培训.Java培训.期待与您交流! ---------- [内存管理] 1.内存管理概念 由于移动设备内存及其有限,所以每个app所占的内存也是有限的 需要回收一些不使用的空间 2.OC内存管理的范围 管理任何继承NSOject的对象,对其他的基本数据类型无效 主要管理堆区中的对象的内存管理   3.内存管理的原理 1)对象所有权概念 任何对象都可以能拥有一个或多个所有者,只要一个对象至少还拥有一个所有者,他就会继续存在 cocoasu所有权策略 任何自

【iOS开发-33】学习手动内存管理临时抛弃ARC以及retain/assign知识——iOSproject师面试必考内容

我们为什么须要内存管理?当使用内存达到40M和45M时候会发出警告,假设不处理,占用内存达到120M时直接强制关闭程序. 所以出现闪退除了是程序出现逻辑错误,还有可能是内存使用过大. (1)创建一个对象的过程:先分配内存空间存储对象:初始化成员变量:返回对象的指针. (2)对象在创建时,内部会自己主动创建一个引用计数器retainCount,当retainCount=0时,系统会回收当前对象,retainCount是唯一推断标记.release会-1.retain会+1,retain后返回的是自

小白学开发(iOS)OC_手动内存管理(2015-08-02)

// //  main.m //  手动内存管理 // //  Created by admin on 15/8/3. //  Copyright (c) 2015年 admin. All rights reserved. // #import <Foundation/Foundation.h> #import "Person.h" int main(int argc, const char * argv[]) { /* 当我们创建一个对象时: 1. 分配内存空间,存储对象

【iOS开发-33】学习手动内存管理暂时抛弃ARC以及retain/assign知识——iOS工程师面试必考内容

我们为什么需要内存管理?当使用内存达到40M和45M时候会发出警告,如果不处理,占用内存达到120M时直接强制关闭程序.所以出现闪退除了是程序出现逻辑错误,还有可能是内存使用过大. (1)创建一个对象的过程:先分配内存空间存储对象:初始化成员变量:返回对象的指针. (2)对象在创建时,内部会自动创建一个引用计数器retainCount,当retainCount=0时,系统会回收当前对象,retainCount是唯一判断标记.release会-1,retain会+1,retain后返回的是自己se

IOS基础 Day-1手动内存管理

辞职回家打算自学IOS开发,就在借个地方记录一下 Day-1      手动内存管理                   主要内容:release  retain必须配对好,不然会占用内存 慢慢积累导致错误 一旦内存占用超过40m 45m时分别发生警告,一旦超过120m 系统将kill你的app 发生闪退 主要要防止发生的问题: 1.野指针操作 2.内存泄漏 理解retain和assign的区别和 retain的原理 Main 1 // 2 // main.m 3 // 1-1内存管理 4 //