强引用(强持有)

简介: 强引用(强持有)

强引用(强持有)


解决NSTimer强引用:


  • 重写didMoveToParentViewController方法


- (void)didMoveToParentViewController:(UIViewController *)parent{  
    // 无论push 进来 还是 pop 出去 正常跑  
    // 就算继续push 到下一层 pop 回去还是继续  
    if (parent == nil) {  
       [self.timer invalidate];  
        self.timer = nil;  
        NSLog(@"timer 走了");  
    }  
}


  • 定义timer时,采用闭包的形式,因此不需要指定target


- (void)blockTimer{  
    self.timer = [NSTimer scheduledTimerWithTimeInterval:1 repeats:YES block:^(NSTimer * _Nonnull timer) {  
        NSLog(@"timer fire - %@",timer);  
    }];  
}


解决 强引用(强持有)


依赖中介者模式,打破强持有,其中推荐思路四


1. pop时在其他方法中销毁timer


- (void)didMoveToParentViewController:(UIViewController *)parent{  
    // 无论push 进来 还是 pop 出去 正常跑  
    // 就算继续push 到下一层 pop 回去还是继续  
    if (parent == nil) {  
       [self.timer invalidate];  
        self.timer = nil;  
        NSLog(@"timer 走了");  
    }  
}


2.中介者模式,即不使用self,依赖于其他对象


//**********1、定义其他对象**********  
@property (nonatomic, strong) id            target;  
//**********1、修改target**********  
self.target = [[NSObject alloc] init];  
class_addMethod([NSObject class], @selector(fireHome), (IMP)fireHomeObjc, "v@:");  
self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self.target selector:@selector(fireHome) userInfo:nil repeats:YES];  
//**********3、imp**********  
void fireHomeObjc(id obj){  
    NSLog(@"%s -- %@",__func__,obj);  
}  
- (void)dealloc{  
    [self.timer invalidate];  
    self.timer = nil;  
    NSLog(@"%s",__func__);  
}


3.自定义封装timer


//*********** .h文件 ***********  
@interface CJLTimerWapper : NSObject  
- (instancetype)cjl_initWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;  
- (void)cjl_invalidate;  
@end  
//*********** .m文件 ***********  
#import "CJLTimerWapper.h"  
#import <objc/message.h>  
@interface CJLTimerWapper ()  
@property(nonatomic, weak) id target;  
@property(nonatomic, assign) SEL aSelector;  
@property(nonatomic, strong) NSTimer *timer;  
@end  
@implementation CJLTimerWapper  
- (instancetype)cjl_initWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo{  
    if (self == [super init]) {  
        //传入vc  
        self.target = aTarget;  
        //传入的定时器方法  
        self.aSelector = aSelector;  
        if ([self.target respondsToSelector:self.aSelector]) {  
            Method method = class_getInstanceMethod([self.target class], aSelector);  
            const char *type = method_getTypeEncoding(method);  
            //给timerWapper添加方法  
            class_addMethod([self class], aSelector, (IMP)fireHomeWapper, type);  
            //启动一个timer,target是self,即监听自己  
            self.timer = [NSTimer scheduledTimerWithTimeInterval:ti target:self selector:aSelector userInfo:userInfo repeats:yesOrNo];  
        }  
    }  
    return self;  
}  
//一直跑runloop  
void fireHomeWapper(CJLTimerWapper *wapper){  
    //判断target是否存在  
    if (wapper.target) {  
        //如果存在则需要让vc知道,即向传入的target发送selector消息,并将此时的timer参数也一并传入,所以vc就可以得知`fireHome`方法,就这事这种方式定时器方法能够执行的原因
        //objc_msgSend发送消息,执行定时器方法  
        void (*lg_msgSend)(void *,SEL, id) = (void *)objc_msgSend;  
         lg_msgSend((__bridge void *)(wapper.target), wapper.aSelector,wapper.timer);  
    }else{  
        //如果target不存在,已经释放了,则释放当前的timerWrapper  
        [wapper.timer invalidate];  
        wapper.timer = nil;  
    }  
}  
//在vc的dealloc方法中调用,通过vc释放,从而让timer释放  
- (void)cjl_invalidate{  
    [self.timer invalidate];  
    self.timer = nil;  
}  
- (void)dealloc  
{
    NSLog(@"%s",__func__);  
}  
@end


4. 利用NSProxy虚基类的子类


//************NSProxy子类************  
@interface CJLProxy : NSProxy  
+ (instancetype)proxyWithTransformObject:(id)object;  
@end  
@interface CJLProxy()  
@property (nonatomic, weak) id object;  
@end  
@implementation CJLProxy  
+ (instancetype)proxyWithTransformObject:(id)object{  
    CJLProxy *proxy = [CJLProxy alloc];  
    proxy.object = object;  
    return proxy;  
}  
-(id)forwardingTargetForSelector:(SEL)aSelector {  
    return self.object;     
}
//************解决timer强持有问题************  
self.proxy = [CJLProxy proxyWithTransformObject:self];  
self.timer = [NSTimer scheduledTimerWithTimeInterval:1 target:self.proxy selector:@selector(fireHome) userInfo:nil repeats:YES];  
//在dealloc中将timer正常释放  
- (void)dealloc{  
    [self.timer invalidate];  
    self.timer = nil;  
}




目录
相关文章
|
2月前
|
存储 Java 程序员
深入理解ThreadLocal及其弱引用
深入理解ThreadLocal及其弱引用
|
缓存 Java 程序员
Java 中的强引用、弱引用、软引用和虚引用
引用方式内存不足时垃圾回收机制启动时其余情况强引用否否否软引用是否否弱引用是是否虚引用可能可能可能在一般的程序中,弱引用和虚引用很少会被用到,强引用天天都在用就不必我说了,在有些情况下(如资源文件很大,但并不是全部要在同一时刻使用)我们会使用软引用来代替原来的强引用,一是可以避免某些情况下产生的 OOM,此外,它可以减少程序使用的内存,加速程序的运行。
107 1
|
8月前
|
Java
【JVM】深入理解Java引用类型:强引用、软引用、弱引用和虚引用
【JVM】深入理解Java引用类型:强引用、软引用、弱引用和虚引用
539 0
|
8月前
|
缓存 Java 程序员
Java垃圾回收: 什么是强引用、软引用、弱引用和虚引用?
Java垃圾回收: 什么是强引用、软引用、弱引用和虚引用?
82 2
|
Java
JVM学习日志(八) 强引用,软引用,弱引用,虚引用
强引用,软引用,弱引用,虚引用 简述
108 0
JVM学习日志(八) 强引用,软引用,弱引用,虚引用
|
算法 Java C++
14-理解Java中的不同引用类型:强引用、软引用、弱引用和虚引用
这篇文章将深入探讨Java中的四种引用类型:强引用、软引用、弱引用和虚引用。我们将逐一解释每种引用类型的特性和用途。
126 0
14-理解Java中的不同引用类型:强引用、软引用、弱引用和虚引用
|
Java
强引用、软引用、弱引用、虚引用的区别?
强引用、软引用、弱引用、虚引用的区别?
89 0
|
Java C++
Java引用类型:强引用,软引用,弱引用,虚引用
Java引用类型:强引用,软引用,弱引用,虚引用
90 0
|
缓存 Java 关系型数据库
强引用、软引用、弱引用、幻象引用有什么区别和使用场景
强引用、软引用、弱引用、幻象引用有什么区别和使用场景
227 1
|
算法 Java
java中的强引用、软引用、弱引用、虚引用
java中将引用类型分为强引用、软引用、弱引用、虚引用。之所以要这么划分,还是为了GC时更好的对对象进行处理。因为jvm已经明确了各种引用的GC方式,所以谈四种引用最好是和GC一起聊才会有意义(个人愚见)。
150 0

热门文章

最新文章