iOS NSTimer

2019-04-01  本文已影响0人  YANGXIXIYear

前段时间,做了一个视频播放的功能,用到了NSTimer,测试时,发现会出现在退出播放的界面或退到后台的时候,还会有播放的声音,也就是说定时器停止的功能失效,这里解析一下Timer无法释放(循环引用)的问题。

_timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(timerMethod) userInfo:nil repeats:YES];
- (void)dealloc {
    [_timer invalidate];
    _timer = nil;
}

如上代码,运行发现,并没有走dealloc方法,发生了内存泄漏,原因是self强引用Timer,而当前Timer也强引用target---self,那么要打破循环引用,有什么办法呢???

方法1:在界面消失时释放定时器

- (void)didMoveToParentViewController:(UIViewController *)parent {
    /**
     *  当从一个视图控制容器中添加或者移除viewController后,该方法被调用。
     *  parent:父视图控制器,如果没有父视图控制器,将为nil
     */
    if (!parent) {
        NSLog(@"Timer move to last");
        [_timer invalidate];
        _timer = nil;
 } 
}

方法2:引入中间者

// 定义一个中间变量target
_timerTarget = [NSObject new];
/** Runtime 给target动态添加方法
 *   [_timerTarget class]: 表示给timerTarget所在的类添加方法
 *   @selector(timerMethod):表示添加的方法
 *   class_getMethodImplementation([self class], @selector(timerMethod)):表示方法的实现(函数 => 函数入口 =>  函数名)
*    "v@:":方法类型(void用v来表示,id用参数@来表示,SEL用:来表示)
 */
class_addMethod([_timerTarget class], @selector(timerMethod), class_getMethodImplementation([self class], @selector(timerMethod)), "v@:");
// 定义timer,设置target
_timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target: _timerTarget selector:@selector(timerMethod) userInfo:nil repeats:YES];

此时会正常走dealloc方法

方法3:引入NSProxy,消息转发

1> 创建一个继承自NSProxy的类:

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface CAProxy : NSProxy
/**
 *  消息转发机制 真正的target
 */
@property (nonatomic, weak) id target;

@end

NS_ASSUME_NONNULL_END
#import "CAProxy.h"

@implementation CAProxy

- (NSMethodSignature *)methodSignatureForSelector:(SEL)sel {
    /**
     *  找到方法签名
     */
    return [self.target methodSignatureForSelector:sel];
}

- (void)forwardInvocation:(NSInvocation *)invocation {
    /**
     *  设置 当前的消息执行者 为target
     */
    [invocation invokeWithTarget:self.target];

@end

2> 使用:

_caProxy = [CAProxy alloc]; // 只有alloc 没有init方法
_caProxy.target = self; // weak持有
// target 设为_caProxy,而不是_caProxy.target
_timer = [NSTimer scheduledTimerWithTimeInterval:1.0f target:_caProxy selector:@selector(timerMethod) userInfo:nil repeats:YES]; 

3> 运行正常走dealloc方法

方法4:iOS10.0后,用block完成

__weak typeof(self) weakSelf = self;
_timer = [NSTimer scheduledTimerWithTimeInterval:1.0 repeats:YES block:^(NSTimer * _Nonnull timer) {
        __strong typeof(weakSelf) strongSelf = weakSelf;
        [strongSelf timerMethod];
    }];

运行正常走dealloc方法

方法5:参照方法4原生方法,给NSTimer添加分类,即可用于iOS10以前

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface NSTimer (CACustomTimer)

+ (NSTimer *)ca_scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block;

@end

NS_ASSUME_NONNULL_END
#import "NSTimer+CACustomTimer.h"

@implementation NSTimer (CACustomTimer)

+ (NSTimer *)ca_scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer * _Nonnull))block {
    
    return [self scheduledTimerWithTimeInterval:interval target:self selector:@selector(ca_blockHandle:) userInfo:block repeats:repeats];
}

+ (void)ca_blockHandle:(NSTimer *)timer {
    void(^block)(void) = timer.userInfo;
    if (block) {
        block();
    }   
}
@end

调用

__weak typeof(self) weakSelf = self;
_timer = [NSTimer ca_scheduledTimerWithTimeInterval:1.0 repeats:YES block:^(NSTimer * _Nonnull timer) {
        __strong typeof(weakSelf) strongSelf = weakSelf; 
        [strongSelf timerMethod];
    }];
上一篇下一篇

猜你喜欢

热点阅读