91超碰碰碰碰久久久久久综合_超碰av人澡人澡人澡人澡人掠_国产黄大片在线观看画质优化_txt小说免费全本

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

iOS中怎樣解決NSTimer的循環引用

發布時間:2021-02-20 10:54:27 來源:億速云 閱讀:197 作者:小新 欄目:移動開發

小編給大家分享一下iOS中怎樣解決NSTimer的循環引用,相信大部分人都還不怎么了解,因此分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后大有收獲,下面讓我們一起去了解一下吧!

一 發現問題

我們都知道NSTimer采用target-action的方式,通常target又是類本身,我們為了方便又把NSTimer聲明為屬性變量,這樣就難免會造成循環引用(需要反復執行計時任務時,如果是單次的任務就不會造成循環引用)。

例如:

_timer = [NSTimer scheduledTimerWithTimeInterval:5.0
           target:self
selector:@selector(startTimer) userInfo:nil
repeats:YES];

深入理解,類有一個成員變量_timer,給_timer設置的target為這個類本身。這樣類保留_timer,_timer又保留了這個類,就會出現循環引用的問題,最后導致類無法正確釋放。

大家可能覺得解決這個問題很簡單,在合適的時機釋放NSTimer,大多人多會選擇viewWillDisappear,viewDidDisappear,dealloc。當然了如果選擇在dealloc釋放NSTimer的且覺得這樣沒問題的,那是你不夠了解dealloc的執行時間,科普下dealloc的執行時機是在self釋放之后執行的。這樣就排除了dealloc,那就只能選擇viewWillDisappear,viewDidDisappear(push和pop都會執行)。但是這兩個方法往往不能滿足需求。

二 解決問題

有去了解NSTimer循環引用的同學,知道有兩種常見的方法可以解決:

  • 采用block封裝,target設置為NSTimer本身

  • 既然是因為target是self本身造成的,那就把target設置為其他對象

(第一種block就不用說了,大家也比較喜歡這種方式,但是有時候就不想用block呢,想用第二種方法,但是用起來有很多不便之處,target是其他對象,action也要在其他對象,這樣在action想要訪問self的相關信息就很不方便。于是就有第三種方法誕生了。)

3.用一個含有weak屬性的對象A包裹self作為target,再對A進行消息轉發,訪問A就相當于訪問self,這樣就完美的解決了循環引用,且保留了target-action方式。

大家比較好奇的是有weak屬性的對象A的類怎么實現,下面來看看代碼:

#import <Foundation/Foundation.h>
#pragma mark -
#pragma mark - 內置weak對象(可用于分類定義weak屬性)
@interface XWWeakObject : NSObject

@property (nullable, nonatomic, weak, readonly) id weakObject;

- (instancetype _Nullable )initWeakObject:(id _Nullable )obj;

+ (instancetype _Nullable )proxyWeakObject:(id _Nullable )obj;
@end

#import "XWWeakObject.h"
@implementation XWWeakObject
-(instancetype)initWeakObject:(id)obj{
  _weakObject = obj;
  return self;
}

+(instancetype)proxyWeakObject:(id)obj{
  
  return [[XWWeakObject alloc] initWeakObject:obj];
}


- (id)forwardingTargetForSelector:(SEL)selector {
  return _weakObject;
}

- (void)forwardInvocation:(NSInvocation *)invocation {
  void *null = NULL;
  [invocation setReturnValue:&null];
}

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
  return [NSObject instanceMethodSignatureForSelector:@selector(init)];
}

- (BOOL)respondsToSelector:(SEL)aSelector {
  return [_weakObject respondsToSelector:aSelector];
}

- (BOOL)isEqual:(id)object {
  return [_weakObject isEqual:object];
}

- (NSUInteger)hash {
  return [_weakObject hash];
}

- (Class)superclass {
  return [_weakObject superclass];
}

- (Class)class {
  return [_weakObject class];
}

- (BOOL)isKindOfClass:(Class)aClass {
  return [_weakObject isKindOfClass:aClass];
}

- (BOOL)isMemberOfClass:(Class)aClass {
  return [_weakObject isMemberOfClass:aClass];
}

- (BOOL)conformsToProtocol:(Protocol *)aProtocol {
  return [_weakObject conformsToProtocol:aProtocol];
}

- (BOOL)isProxy {
  return YES;
}

- (NSString *)description {
  return [_weakObject description];
}

- (NSString *)debugDescription {
  return [_weakObject debugDescription];
}
@end

XWWeakObject類有一個weak只讀weakObject對象(這個類也可以用于分類聲明weak屬性:分類是本身是不能聲明weak屬性的)。

用運行時對該類的對象做了消息轉發,對象轉發,在訪問XWWeakObject對象的時候相當于訪問其屬性weakObject對象。

最后看下怎么用代碼實現的:

- (void)viewDidLoad {
  [super viewDidLoad];
  XWWeakObject *target = [XWWeakObject proxyWeakObject:self];
  self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:target selector:@selector(timerCount) userInfo:nil repeats:YES];
}

-(void)timerCount{ 
}

-(void)dealloc{
  [_timer invalidate];
   _timer = nil;
}

前提timer是self的一個屬性,創建一個XWWeakObject對象target,target是內部weak屬性指向self,相當于target擁有self且是weak,self的retain沒有加1,timer擁有XWWeakObject對象target,target的retain加1,timer和self的直接關系是timer僅是self的一個屬性,這樣看來并沒有形成循環引用。

三 寫在最后

雖然這種方式沒有block簡便,但不失為一種好的方法,保存了系統的方式。喜歡用target-action方式的或者不太熟悉block的可以學一學哦,且XWWeakObject能做的不僅僅這些,XWWeakObject可以解決很多類似的循環引用問題,解決分類定義weak屬性等等

有人可能有疑問,為什么都同樣是target-action方式button就不會出現循環引用的問題,有去研究的同學應該都知道UIControl的內部做了weak操作,即真正持有的時候是weak的并沒有導致retain加1,而NSTimer由于runloop的原因并沒有做weak操作。

以上是“iOS中怎樣解決NSTimer的循環引用”這篇文章的所有內容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內容對大家有所幫助,如果還想學習更多知識,歡迎關注億速云行業資訊頻道!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

迭部县| 阳泉市| 拉孜县| 抚远县| 茶陵县| 布拖县| 安庆市| 忻州市| 康乐县| 广丰县| 扎囊县| 铜山县| 邹城市| 怀来县| 南安市| 长垣县| 泸水县| 化隆| 宜昌市| 乐东| 哈巴河县| 漯河市| 南岸区| 瑞丽市| 水城县| 神木县| 壤塘县| 四会市| 永嘉县| 泰兴市| 云林县| 中江县| 吕梁市| 普定县| 大余县| 融水| 潜山县| 平罗县| 合肥市| 鄯善县| 连江县|