下面的代码有没有内存泄漏检测工具,如果有,请说明clss student

本帖子已过去太久远了,不再提供回复功能。iOS开发中本人或同事碰到的内存泄漏及解决办法 - 简书
iOS开发中本人或同事碰到的内存泄漏及解决办法
首先需要理解两个概念:
OC内存管理采用的是引用计数的方式(详细请百度)
内存泄漏:为什么要说这个,因为我发现好多人都在说内存泄漏但并没有理解,这里我用大白话讲一下:就是有块内存你虽然不用了但还要占着不让别人用,所以称为内存泄漏(专业的来了:memory leak)
weak关键的作用(请百度)
好的,进入正题,我逐步演示碰到的各种内存泄漏:
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableArray *arr1 = [NSMutableArray array];
NSMutableArray *arr2 = [NSMutableArray array];
[arr1 addObject:arr2];
[arr2 addObject:arr1];
我们用Xcode的Instruments工具检测一下
Paste_Image.png
上图中我们已经很明显的看到了循环引用解决办法:打破环!上述例子中将arr1或者arr2任意一个 弱引用就OK贴张图出来与上面代码的不同会更加清晰(^__^)
再次用工具检测:
看到leak那行都是绿色的是不是很棒?
首先我们建两个model:Student、TeacherStudent类中有属性teacherTeacher类中有属性student并且都重写了dealloc方法,打印信息
#import &Foundation/Foundation.h&
#import "Student.h"
@interface Teacher : NSObject
@property (nonatomic, strong) Student *
#import "Teacher.h"
@implementation Teacher
- (void)dealloc {
NSLog(@"%s",__func__);
#import &Foundation/Foundation.h&
@interface Student : NSObject
@property (nonatomic, strong) Teacher *
#import "Student.h"
@implementation Student
- (void)dealloc {
NSLog(@"%s",__func__);
第一个页面push到第二个页面,第二个页面代码如下:
#import "ViewController2.h"
#import "Student.h"
#import "Teacher.h"
@interface ViewController2 ()
Teacher *_
Student *_
@implementation ViewController2
// dealloc
- (void)dealloc {
NSLog(@"%s",__func__);
- (void)viewDidLoad {
[super viewDidLoad];
_student = [[Student alloc]init];
_teacher = [[Teacher alloc]init];
_student.teacher = _
_teacher.student = _
运行push到第二页面在pop回第一个页面看控制台的打印信息
仅仅走了Controller的dealloc方法,两个model的dealloc都没有走
为什么呢?我们分析一下
觉得还是画图解释的快一点((⊙﹏⊙)虽然小编累一点)如图:上半部分是两个model的UML图,下半部分我们看到_student实例的teacher属性引用了_teacher实例,而实例_teacher的student属性引用了_student实例这样就形成了一个引用环,由于OC的内存管理机制就导致了这两块内存不能被释放导致内存泄漏解决办法:打破环!我们将Student或Teacher类里的属性 任意一个内存语义strong改为weak,再跑一下看看结果:
三个dealloc方法都走了,是不是很nice?
最最常见的一种,block中的操作
typedef void(^TestBlock)(void);
@interface ViewController2 ()
@property (nonatomic, copy) TestB
@implementation ViewController2
- (void)dealloc {
NSLog(@"%s",__func__);
- (void)viewDidLoad {
[super viewDidLoad];
self.block = ^ {
self.view.backgroundColor = [UIColor redColor];
block为什么用copy修饰(请百度,不在本次讨论重点内)由于block会retain当前对象,所以这里也形成了一个环vc2引用block,block保留当前对象self解决办法:打破环将block内的self弱引用就OK
上面3中情况都是循环引用造成的内存泄漏,解决办法都是打破环ok,看如下代码:
- (void)viewDidLoad {
[super viewDidLoad];
NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
NSOperationQueue *mainQuene =[NSOperationQueue mainQueue];
[center addObserverForName:UIKeyboardWillShowNotification
object:nil
queue:mainQuene
usingBlock:^(NSNotification *note) {
self.view.backgroundColor = [UIColor redColor];
当pop到上一页面的时候 dealloc 方法也没有走,why?我们分析一下:center -& block -& self但是self有没有拥有center?答案是有的,具体里边的实现暂时我不清楚,但确定的是有环的存在
解决办法:打破环
__weak typeof(self) weakSelf =
NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
NSOperationQueue *mainQuene =[NSOperationQueue mainQueue];
[center addObserverForName:UIKeyboardWillShowNotification
object:nil
queue:mainQuene
usingBlock:^(NSNotification *note){
weakSelf.view.backgroundColor = [UIColor redColor];
以上就是我碰到的或同事碰到的导致内存泄漏的几种情况,真正开发中碰到的应该都比较复杂,利用好Instruments加上开发时注意保持良好习惯,大都可以避免
其实这篇文章很早就想写,但是这边涉及的内容确实太多了block为什么会retain当前对象?OC为什么采用引用计数来管理内存?管理内存语义的那些关键字(strong,weak,copy等)之间的差异Instruments工具怎么灵活运用?等等,所以本篇就只针对内存泄漏,其他我也不很熟,很多没想通,也不敢写
这下也就很容易理解为什么delegate要用weak修饰了(不明白的话留言,我再补个图上来)so:到此结束!(一气呵成的感觉好爽O(∩_∩)O哈哈~)
希望会给大家带来帮助(^o^)/~
曾梦想给中国IT带来变化

我要回帖

更多关于 内存泄漏的代码 的文章

 

随机推荐