日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

IOS內存泄漏檢查方法及重寫MLeakFinder

瀏覽:31日期:2022-09-16 16:21:09

對于iOS開發來講,內存泄漏的問題,已經是老生常談的話題。在日常的面試中經常會提到這些問題。我們日常的開發過程中進行內存泄漏的檢測,一般是使用instrument工具中的Leaks/Allocation來進行排查,網絡上也有比較高效又好用的內存泄漏檢測工具,MLeakFinder。

MLeakFinder-原理

首先看UIViewController,當一個UIViewController被pop或dismiss的時候,這個VC包括在這個VC上的View,或者子View都會很快的被釋放。所以我們我們需要在UIViewController被POP或dismiss后一小段時間后,在這個VC上的view,subView等是否還存在。

在UIViewController+MemoryLeak.h的load方法中可以看到,早+load方法中通過runtime交換了viewWillAppear,viewDidAppear,dismissViewControllerAnimated:completion:這三個方法。

1,首先看viewWillAppear

- (void)swizzled_viewWillAppear:(BOOL)animated { [self swizzled_viewWillAppear:animated]; objc_setAssociatedObject(self, kHasBeenPoppedKey, @(NO), OBJC_ASSOCIATION_RETAIN);}

當VC進來的時候,添加關聯對象,并標記為NO

2,在看viewDidAppear

- (void)swizzled_viewDidDisappear:(BOOL)animated { [self swizzled_viewDidDisappear:animated]; if ([objc_getAssociatedObject(self, kHasBeenPoppedKey) boolValue]) {[self willDealloc];}}

通過代碼可以看出,獲取當前關聯對象的標記,當標記為YES的時候,就會調用willDealloc。

3,我們看什么時候會被標記為YES呢?

在UINavigationController+MemoryLeak.h的popViewControllerAnimated:方法中我們可以看到

- (UIViewController *)swizzled_popViewControllerAnimated:(BOOL)animated { UIViewController *poppedViewController = [self swizzled_popViewControllerAnimated:animated]; if (!poppedViewController) {return nil; } // Detail VC in UISplitViewController is not dealloced until another detail VC is shown if (self.splitViewController &&self.splitViewController.viewControllers.firstObject == self &&self.splitViewController == poppedViewController.splitViewController) {objc_setAssociatedObject(self, kPoppedDetailVCKey, poppedViewController, OBJC_ASSOCIATION_RETAIN)return poppedViewController; } // VC is not dealloced until disappear when popped using a left-edge swipe gesture extern const void *const kHasBeenPoppedKey; objc_setAssociatedObject(poppedViewController, kHasBeenPoppedKey, @(YES), OBJC_ASSOCIATION_RETAIN); return poppedViewController;}

我們可以看出,在VC被pop或者左滑返回的時候,相當于視圖銷毀,就會被標記為YES。

4,我們重點看willDealloc

- (BOOL)willDealloc { //第一步 NSString *className = NSStringFromClass([self class]); if ([[NSObject classNamesWhitelist] containsObject:className])return NO; //第二步 NSNumber *senderPtr = objc_getAssociatedObject([UIApplication sharedApplication], kLatestSenderKey) if ([senderPtr isEqualToNumber:@((uintptr_t)self)])return NO; //第三步 __weak id weakSelf = self; dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{__strong id strongSelf = weakSelf;[strongSelf assertNotDealloc]; }); return YES;}

1.第一步:我們可以看到,會先判斷當前的class是否在白名單中,是的話就會return NO,即不是內存泄漏的。同時我們查看構建白名單的源碼:使用了一個單例實現,確保只有一個,是個私有方法

+ (NSMutableSet *)classNamesWhitelist { static NSMutableSet *whitelist = nil; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{whitelist = [NSMutableSet setWithObjects: @'UIFieldEditor', // UIAlertControllerTextField @'UINavigationBar', @'_UIAlertControllerActionView', @'_UIVisualEffectBackdropView', nil];// System’s bug since iOS 10 and not fixed yet up to this ci.NSString *systemVersion = [UIDevice currentDevice].systemVersion;if ([systemVersion compare:@'10.0' options:NSNumericSearch] != NSOrderedAscending) { [whitelist addObject:@'UISwitch'];} }); return whitelist;}

同時在還支持,自定義的添加白名單

+ (void)addClassNamesToWhitelist:(NSArray *)classNames { [[self classNamesWhitelist] addObjectsFromArray:classNames];}

2. 第二步:判斷該對象是否是上一次發送action的對象,是的話,不進行內存檢測

//第二步 NSNumber *senderPtr = objc_getAssociatedObject([UIApplication sharedApplication], kLatestSenderKey) if ([senderPtr isEqualToNumber:@((uintptr_t)self)])return NO;

3,第三步:弱指針指向self,2s延遲,然后通過這個弱指針調用-assertNotDealloc,若被釋放,給nil發消息直接返回,不觸發-assertNotDealloc方法,認為已經釋放;如果它沒有被釋放(泄漏了),-assertNotDealloc就會被調用

__weak id weakSelf = self; dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{__strong id strongSelf = weakSelf;[strongSelf assertNotDealloc]; });5,現在我們回到:2的代碼 [self willDealloc]

看一下他的源碼

- (BOOL)willDealloc { //第一步 if (![super willDealloc]) {return NO; } //第二步 [self willReleaseChildren:self.childViewControllers]; [self willReleaseChild:self.presentedViewController]; if (self.isViewLoaded) {[self willReleaseChild:self.view]; } return YES;}

1,第一步:會通過 super調用父類的willDealloc,即上面目錄4

2,第二步:調用willReleaseChildren,willReleaseChild遍歷該對象的子對象,看其是否釋放

- (void)willReleaseChild:(id)child { if (!child) {return; }[self willReleaseChildren:@[ child ]];}- (void)willReleaseChildren:(NSArray *)children { NSArray *viewStack = [self viewStack]; NSSet *parentPtrs = [self parentPtrs]; for (id child in children) {NSString *className = NSStringFromClass([child class]);[child setViewStack:[viewStack arrayByAddingObject:className]];[child setParentPtrs:[parentPtrs setByAddingObject:@((uintptr_t)child)]];[child willDealloc]; }}

通過代碼可以看出,通過調用willReleaseChildren的方法,獲取當前對象viewStack,parentPtrs,并且遍歷children,為每個子對象設置viewStack,parentPtrs,然后調用willDealloc。

通過源碼看一下viewStask,parentPtrs的實現

- (NSArray *)viewStack { NSArray *viewStack = objc_getAssociatedObject(self, kViewStackKey); if (viewStack) {return viewStack; }NSString *className = NSStringFromClass([self class]); return @[ className ];}- (void)setViewStack:(NSArray *)viewStack { objc_setAssociatedObject(self, kViewStackKey, viewStack, OBJC_ASSOCIATION_RETAIN);}- (NSSet *)parentPtrs { NSSet *parentPtrs = objc_getAssociatedObject(self, kParentPtrsKey); if (!parentPtrs) {parentPtrs = [[NSSet alloc] initWithObjects:@((uintptr_t)self), nil]; } return parentPtrs;}- (void)setParentPtrs:(NSSet *)parentPtrs { objc_setAssociatedObject(self, kParentPtrsKey, parentPtrs, OBJC_ASSOCIATION_RETAIN);}

viewStack使用數組,parentPtrs使用的集合形式。都是通過運行時,用關聯對象添加屬性。

parentPtrs會在-assertNotDealloc中,會判斷當前對象是否與父節點集合有交集。下面仔細看下-assertNotDealloc方法

- (void)assertNotDealloc { //第一步 if ([MLeakedObjectProxy isAnyObjectLeakedAtPtrs:[self parentPtrs]]) {return; } //第二步 [MLeakedObjectProxy addLeakedObject:self];NSString *className = NSStringFromClass([self class]); NSLog(@'Possibly Memory Leak.nIn case that %@ should not be dealloced, override -willDealloc in %@ by returning NO.nView-ViewController stack: %@', className, className, [self viewStack]);}

1,第一步我們看到,通過parentPtrs的判斷是否有交集

產看其源碼:

+ (BOOL)isAnyObjectLeakedAtPtrs:(NSSet *)ptrs { NSAssert([NSThread isMainThread], @'Must be in main thread.'); static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{leakedObjectPtrs = [[NSMutableSet alloc] init]; }); if (!ptrs.count) {return NO } if ([leakedObjectPtrs intersectsSet:ptrs]) {return YES; } else {return NO; }}

可以看到,創建了一個單例對象,通過集合的形式,判斷是否有交集,是的話return。否則就進入第二步

2,第二步:addLeakedObject

+ (void)addLeakedObject:(id)object { NSAssert([NSThread isMainThread], @'Must be in main thread.'); MLeakedObjectProxy *proxy = [[MLeakedObjectProxy alloc] init]; proxy.object = object; proxy.objectPtr = @((uintptr_t)object); proxy.viewStack = [object viewStack]; static const void *const kLeakedObjectProxyKey = &kLeakedObjectProxyKey; objc_setAssociatedObject(object, kLeakedObjectProxyKey, proxy, OBJC_ASSOCIATION_RETAIN); [leakedObjectPtrs addObject:proxy.objectPtr];#if _INTERNAL_MLF_RC_ENABLED [MLeaksMessenger alertWithTitle:@'Memory Leak' message:[NSString stringWithFormat:@'%@', proxy.viewStack] delegate:proxy additionalButtonTitle:@'Retain Cycle'];#else [MLeaksMessenger alertWithTitle:@'Memory Leak' message:[NSString stringWithFormat:@'%@', proxy.viewStack]];#endif}

第一步:構造MLeakedObjectProxy對象,給傳入的泄漏對象 object 關聯一個代理即 proxy

第二步:通過objc_setAssociatedObject(object, kLeakedObjectProxyKey, proxy, OBJC_ASSOCIATION_RETAIN)方法,object強持有proxy, proxy若持有object,如果object釋放,proxy也會釋放

第三步:存儲 proxy.objectPtr(實際是對象地址)到集合 leakedObjectPtrs 里邊

第四步:彈框 AlertView若 _INTERNAL_MLF_RC_ENABLED == 1,則彈框會增加檢測循環引用的選項;若 _INTERNAL_MLF_RC_ENABLED == 0,則僅展示堆棧信息。

對于MLeakedObjectProxy類而言,是檢測到內存泄漏才產生的,作為泄漏對象的屬性存在的,如果泄漏的對象被釋放,那么MLeakedObjectProxy也會被釋放,則調用-dealloc函數

集合leakedObjectPtrs中移除該對象地址,同時再次彈窗,提示該對象已經釋放了

6,自己也在嘗試重寫該框架,歡迎大家一起交流

以上就是IOS內存泄漏檢查方法及重寫MLeakFinder的詳細內容,更多關于IOS內存泄漏的資料請關注好吧啦網其它相關文章!

標簽: IOS
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
偷拍亚洲精品| 国产成人免费精品| 美女视频黄免费的久久| 日韩av网站在线观看| 亚洲欧洲专区| 国产欧美日韩精品一区二区免费| 成人在线网站| 国产一区日韩一区| 欧美 日韩 国产一区二区在线视频| av在线最新| 欧美福利在线| 视频一区欧美精品| 91国内精品| 精品视频亚洲| 国产一区日韩欧美| 婷婷亚洲综合| 综合一区在线| 久久精品国产亚洲aⅴ| 成人亚洲精品| 美日韩精品视频| 久久爱www成人| 欧美日韩精品免费观看视完整 | 精品国产亚洲一区二区三区在线 | 欧美激情福利| 久久婷婷激情| 欧美亚洲综合视频| 日韩理论片av| 亚洲区欧美区| 神马久久午夜| 亚洲三区欧美一区国产二区| 免费在线观看一区| 影音先锋久久| 国产精品天天看天天狠| 欧美日韩在线网站| 国产日韩欧美在线播放不卡| 国产精品亚洲一区二区三区在线观看| 亚洲免费成人| 欧美黄页在线免费观看| 精品久久精品| 电影91久久久| 日韩中文字幕亚洲一区二区va在线 | 欧美成人精品一级| 男女男精品视频网| 久久精品二区三区| 精品亚洲精品| 天堂va蜜桃一区二区三区| 久久精品国产一区二区| 日韩精品亚洲一区二区三区免费| 亚洲第一精品影视| 国产中文在线播放| 精品三级久久久| 欧美日本不卡| 日韩毛片一区| 美女久久网站| 免播放器亚洲一区| 国产精品普通话对白| 蜜臀av免费一区二区三区| 精品三级久久| 99国产精品免费视频观看| 国产成人精品福利| 久久爱www.| 久久免费福利| 国产成人在线中文字幕| 另类综合日韩欧美亚洲| 国产精品白浆| 国产一区福利| 亚洲综合电影| 亚洲高清久久| 亚洲精品va| 蜜臀久久99精品久久久久宅男| 免费观看在线综合| 日韩一区免费| 精品中文在线| 久久高清精品| 日韩在线卡一卡二| 亚洲一二av| 91精品国产自产在线丝袜啪| 欧美伊人久久| 国产一区二区三区不卡视频网站 | 日韩福利一区| 五月天久久777| 久久国产高清| 国产精品一区二区免费福利视频| 麻豆精品在线观看| 久久久久免费av| 免费观看日韩电影| 精品久久中文| 蜜桃av一区| 激情黄产视频在线免费观看| 国产精品99一区二区| 日本aⅴ免费视频一区二区三区| 久久麻豆视频| 亚洲三级在线| 视频一区中文| 国产日韩免费| 99久久精品费精品国产| 亚洲一区二区三区中文字幕在线观看| 国产精品亚洲产品| 欧美精品一卡| 久久99高清| 在线国产日韩| 久久精品高清| 欧美激情视频一区二区三区在线播放| 成人免费电影网址| 你懂的亚洲视频| 亚洲va久久| 狠狠色狠狠色综合日日tαg| 国产中文字幕一区二区三区| 亚洲精品大片| 欧美午夜不卡影院在线观看完整版免费| 国产精品nxnn| 日韩在线观看一区二区| 亚洲大片在线| 久久人人88| 国产调教精品| 欧美成人亚洲| 欧美精品导航| 国产精品一级| 影音先锋久久| 久久久人人人| 国产专区精品| 麻豆高清免费国产一区| 伊人久久一区| 国产精品外国| 亚洲一区观看| 男人天堂欧美日韩| 日韩国产欧美| 欧美激情另类| 不卡一二三区| 日韩精品影视| 99久久99久久精品国产片果冰| 日韩黄色大片| 久久婷婷一区| 午夜精品亚洲| 中文字幕亚洲影视| 日韩av一区二| 日本在线一区二区三区| 日本欧美韩国一区三区| 日韩激情中文字幕| 欧美日韩网址| 国产成人调教视频在线观看| 精品国产日韩欧美精品国产欧美日韩一区二区三区 | 久久精品资源| 日韩国产网站| 精品日韩毛片| 亚洲精品高潮| 国产精品a久久久久| 国产精品白丝久久av网站| 日韩电影免费网站| 美女黄网久久| 麻豆一区二区三| 久久九九精品| 日韩精品一区二区三区免费视频| 久久精品99国产精品| 国内精品麻豆美女在线播放视频| 日韩精品中文字幕第1页| 午夜在线精品偷拍| 久久av导航| 亚洲一级特黄| 日韩av二区在线播放| av一区在线| 欧美中文一区| 91欧美日韩| 中文字幕乱码亚洲无线精品一区| 国产亚洲高清在线观看| 99久精品视频在线观看视频| 91成人在线| japanese国产精品| 国产精品宾馆| 亚洲精品女人| 久久国产毛片| 国产一区调教| 国产剧情一区| 在线免费观看亚洲| 日韩网站中文字幕| 久久av免费看| 亚洲精品大全| 久久久夜精品| 精品免费视频| 国产亚洲精品美女久久久久久久久久| 99热精品久久| 色天使综合视频| 国产成人调教视频在线观看| 国产欧美一区二区色老头| 蜜桃伊人久久| 在线亚洲欧美| 午夜影院欧美| 久久久久久久久99精品大| 久久久久久网| 神马午夜在线视频| 最新中文字幕在线播放| 麻豆国产精品视频| 日韩高清不卡在线| 蜜臀精品久久久久久蜜臀| 9国产精品视频| 天使萌一区二区三区免费观看| 欧美成人亚洲| 免费国产亚洲视频| 日韩黄色在线观看| 欧美一区影院|