由于NSTimer會保留target的引用計(jì)數(shù)困檩,所以使用過程中很容易造成循環(huán)應(yīng)用异逐。解決辦法很容易汉嗽,通常會利用block,來實(shí)現(xiàn)self的weak化问潭,從而避免self和target相互強(qiáng)引用猿诸。具體實(shí)現(xiàn)思路和方式,參見
用Block解決NSTimer循環(huán)引用
實(shí)際上狡忙,iOS10的sdk中梳虽,官方已經(jīng)提供了block實(shí)現(xiàn)timer的接口:
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;
另一種思路:消息轉(zhuǎn)發(fā)
循環(huán)引用本質(zhì)在于相互強(qiáng)引用,無法自主釋放灾茁。所以防止循環(huán)產(chǎn)生窜觉,避免閉環(huán)產(chǎn)生即可。
如圖删顶,self強(qiáng)引用timer竖螃,timer強(qiáng)引用weakobject,但weakobject弱引用self逗余,即避免了強(qiáng)引用閉環(huán)的產(chǎn)生特咆。當(dāng)外部不在強(qiáng)引用self,內(nèi)部timer也不會強(qiáng)引用self录粱,引用計(jì)數(shù)為0腻格,self可以正常釋放。
這里要實(shí)現(xiàn)的啥繁,是如何使對weakobject的發(fā)送的消息菜职,轉(zhuǎn)化為self對象來實(shí)現(xiàn)。
//.h file
@interface HACWeakObject : HACObject
@property (nonatomic, weak, readonly) HACObject *target;
+ (instancetype)weakObjectWithTarget:(HACObject*)target ;
@end
//.m file
#import "HACWeakObject.h"
@interface HACWeakObject ()
@property (nonatomic, weak, readwrite) HACObject *target;
@end
@implementation HACWeakObject
+ (instancetype)weakObjectWithTarget:(HACObject *)target {
HACWeakObject *weakObject = [[HACWeakObject alloc] init];
weakObject.target = target;
return weakObject;
}
- (id)forwardingTargetForSelector:(SEL)aSelector {
if (self.target&&[self.target respondsToSelector:aSelector]) {
return self.target;
}
return nil;
}
@end
//use case:
self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:[HACWeakObject weakObjectWithTarget:self] selector:@selector(timerFired:) userInfo:nil repeats:YES];
??代碼旗闽,利用message forward的方式酬核,實(shí)現(xiàn)了weakobject轉(zhuǎn)發(fā)消息到target來實(shí)現(xiàn)的過程。當(dāng)在timer對target适室,即weakobject發(fā)送timerFired消息嫡意,weakobject試圖處理消息,但沒有實(shí)現(xiàn)方法所以觸發(fā)消息轉(zhuǎn)發(fā)捣辆,首先會+resolveInstanceMethod蔬螟,weakobject沒實(shí)現(xiàn),返回NO汽畴,進(jìn)入到forwardingTargetForSelector旧巾,試圖轉(zhuǎn)發(fā)到其他對象解決耸序。在這里設(shè)置處理對象為weak方式保留的self,轉(zhuǎn)到self來響應(yīng)該消息鲁猩。從而實(shí)現(xiàn)既不保留self坎怪,又能夠讓self處理消息的初衷。
消息轉(zhuǎn)發(fā)機(jī)制可以參考:
輕松學(xué)習(xí)之 Objective-C消息轉(zhuǎn)發(fā)
補(bǔ)充:
最新發(fā)現(xiàn)廓握,即使self的assign屬性芋忿,如NSInteger類型,當(dāng)使用block方式疾棵,在block中調(diào)用時(shí),也無法及時(shí)釋放痹仙,產(chǎn)生循環(huán)引用是尔。而assign類型不屬于object,無法通過weak方式解決开仰。目前的解決方案是__block后再block內(nèi)使用拟枚,或者改為NSNumber。