每次项目编译完成以后,都被内存搞得头昏脑胀,压力甚大。git
利用两周时间,稍微研究了 微信开源的 MLeaksFinder 和 facebook 开源的 FBMemoryProfiler,github
这两个开源三方,在编写过程当中就能够检测内存泄露,实在是不要太方便……微信
但愿本身在下一个项目能用的驾轻就熟……ide
1. 微信 MLeaksFiner atom
若是对它,你仍是不是很了解 ,能够到查阅这里: http://wereadteam.github.io/2016/07/20/MLeaksFinder2/ ;spa
1.1 建立demo 项目 LeakDemo 引入 MLeaksFiner 开源文件夹debug
->本次 我是直接下载到本地了,因此直接拖拽到了项目3d
1.2 引入这个开源文件后 ,不用写多余的代码,默认debug 编译的时候,自动启动使用orm
下面来写一段内存泄露的代码,进行试验::对象
内存泄露的代码::
在 main.storybord 中添加 NavigationController
VC 代码 ViewController .m ::
#import "ViewController.h" #import "FirstController.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; self.navigationItem.title = @"vc"; [self creatButton]; } - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; } -(void)creatButton{ UIButton *button = [[UIButton alloc]initWithFrame:CGRectMake(0, 0, 100, 100)]; [self.view addSubview:button]; button.center = self.view.center; [button setTitle:@"下一页" forState:UIControlStateNormal]; button.backgroundColor = [UIColor yellowColor]; [button addTarget:self action:@selector(didButtonClick:) forControlEvents:UIControlEventTouchUpInside]; } -(void)didButtonClick:(UIButton*)btn{ NSLog(@"vc btn Click"); [self.navigationController pushViewController:[FirstController new] animated:true]; }
新建 FirstController :
#import "FirstController.h" #import "TestView.h" @interface FirstController () @property (nonatomic,strong)TestView *testView; @end @implementation FirstController - (void)viewDidLoad { [super viewDidLoad]; self.navigationItem.title = @"first vc"; self.view.backgroundColor = [UIColor whiteColor]; [self addTestView]; } - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; } -(void)addTestView{ self.testView = [[TestView alloc]initWithFrame:CGRectMake(0, 0, 200, 30)]; self.testView.vc = self; [self.view addSubview:self.testView]; self.testView.center = self.view.center; }
@end
新建 TestView :
TestView .h
#import <UIKit/UIKit.h>
#import "FirstController.h"
@interface TestView : UIView
@property (nonatomic,strong)FirstController *vc;// 把这个属性 换成 strong 属性 故意形成 泄露
@end
TestView .m
#import "TestView.h" @implementation TestView - (instancetype)initWithFrame:(CGRect)frame { self = [super initWithFrame:frame]; if (self) { [self prepareUI]; } return self; } -(void)prepareUI{ self.backgroundColor = [UIColor redColor]; } @end
内存泄露的解释::::
vc -> push -> first VC
:( first vc 强持有 testView ,testView 强持有 firstVc : 彼此强持有,没法释放)
-> pop 回-> vc
=》: first vc 没法释放
1.3 检测 到泄露 alert 方式体提醒 :::
这里检测到是有 循环泄露的,可是没法准肯定位泄露的对象
因此继续……
2 . MLeakFinder + FBRetainCycleDetector
2.1 查阅了 MLeaksFinder.h 的文件
这段宏定义的解释 https://www.jianshu.com/p/e3dbf58982f6:::
#ifdef MEMORY_LEAKS_FINDER_ENABLED //_INTERNAL_MLF_ENABLED 宏用来控制 MLLeaksFinder库 //何时开启检测,能够自定义这个时机,默认则是在DEBUG模式下会启动,RELEASE模式下不启动 //它是经过预编译来实现的
#define _INTERNAL_MLF_ENABLED MEMORY_LEAKS_FINDER_ENABLED #else #define _INTERNAL_MLF_ENABLED DEBUG #endif
//_INTERNAL_MLF_RC_ENABLED 宏用来控制 是否开启循环引用的检测
#ifdef MEMORY_LEAKS_FINDER_RETAIN_CYCLE_ENABLED #define _INTERNAL_MLF_RC_ENABLED MEMORY_LEAKS_FINDER_RETAIN_CYCLE_ENABLED //COCOAPODS 由于MLLeaksFinder引用了第三库用来检查循环引用,因此必须是当前项目中使用了COCOAP //ODS,才能使用这个功能。 #elif COCOAPODS #define _INTERNAL_MLF_RC_ENABLED COCOAPODS #endif
2.2 . 添加cocopod ,而后 Pod 添加 FBRetainCycleDetector;
注释 这两段代码 == 》同时开启 MLeakfinder 检测 和 retainCycle 检测
//#define MEMORY_LEAKS_FINDER_ENABLED 0 #ifdef MEMORY_LEAKS_FINDER_ENABLED #define _INTERNAL_MLF_ENABLED MEMORY_LEAKS_FINDER_ENABLED #else #define _INTERNAL_MLF_ENABLED DEBUG #endif //#define MEMORY_LEAKS_FINDER_RETAIN_CYCLE_ENABLED 0 #ifdef MEMORY_LEAKS_FINDER_RETAIN_CYCLE_ENABLED #define _INTERNAL_MLF_RC_ENABLED MEMORY_LEAKS_FINDER_RETAIN_CYCLE_ENABLED #elif COCOAPODS #define _INTERNAL_MLF_RC_ENABLED COCOAPODS #endif
从新编译运行 弹框是这样
正确找到 循环引用 cycle
最简单的使用方法了吧,先使用着,有机会会研究下 源码 ^ V ^