美文网首页
2019-03-07 对WKWebView中的循环引用的理解

2019-03-07 对WKWebView中的循环引用的理解

作者: 幸福晓杰2016 | 来源:发表于2019-03-07 15:32 被阅读0次

    WKWebView的这个循环引用很像定时器的循环引用。
    定时器的循环引用是因为系统runloop循环 强引用了执行定时器方法的对象。
    WKWebView,我认识是WebKit中有关js消息的某个系统 强引用了接收消息的对象。
    我画个逻辑图如下:


    内存不能释放的引用图

    既然知道原因了,我们可以像处理定时器一样,中间给它加一个弱引用,避开这种情况。
    如下图所示:


    加个中间层就能释放的原理图

    2019年08月14日15:25:39补充

    YYKit里面有个类YYWeakProxy,它利用代理机制与消息转发机制,真正的实现了这个中介者去弱引用的思想:
    代码如下:

    #import <Foundation/Foundation.h>
    
    /**
     A proxy used to hold a weak object.
     It can be used to avoid retain cycles, such as the target in NSTimer or CADisplayLink.
     
     sample code:
     
         @implementation MyView {
            NSTimer *_timer;
         }
         
         - (void)initTimer {
            YYWeakProxy *proxy = [YYWeakProxy proxyWithTarget:self];
            _timer = [NSTimer timerWithTimeInterval:0.1 target:proxy selector:@selector(tick:) userInfo:nil repeats:YES];
         }
         
         - (void)tick:(NSTimer *)timer {...}
         @end
     */
    @interface YYWeakProxy : NSProxy
    
    /**
     The proxy target.
     */
    @property (nullable, nonatomic, weak, readonly) id target;
    
    /**
     Creates a new weak proxy for target.
     
     @param target Target object.
     
     @return A new proxy object.
     */
    - (instancetype)initWithTarget:(id)target;
    
    /**
     Creates a new weak proxy for target.
     
     @param target Target object.
     
     @return A new proxy object.
     */
    + (instancetype)proxyWithTarget:(id)target;
    
    @end
    
    
    //
    //  YYWeakProxy.m
    //  YYKit <https://github.com/ibireme/YYKit>
    //
    //  Created by ibireme on 14/10/18.
    //  Copyright (c) 2015 ibireme.
    //
    //  This source code is licensed under the MIT-style license found in the
    //  LICENSE file in the root directory of this source tree.
    //
    
    #import "YYWeakProxy.h"
    
    /**
        实现的原理: 使用 NSProxy 持有 NSTimer 的 target
        不再用 NSTimer 直接持有 self,就不会导致 timer 对 self 的循环强引用了
     */
    @implementation YYWeakProxy
    
    - (instancetype)initWithTarget:(id)target {
        _target = target;
        return self;
    }
    
    //类方法
    + (instancetype)proxyWithTarget:(id)target {
        return [[YYWeakProxy alloc] initWithTarget:target];
    }
    
    #pragma mark - private
    // 在处理消息转发时,将消息转发给真正的Target处理  
    - (id)forwardingTargetForSelector:(SEL)selector {
        return _target;
    }
    
    #pragma mark - over write
    //重写NSProxy如下两个方法,保证代理执行时不会报错
    // 方法签名,甭管传入什么方法,都是转成NSObject的init的方法签名
    //这个函数本意是让重载方有机会抛出一个函数的签名,再由后面的forwardInvocation:去执行为给定消息提供参数类型信息
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
        return [NSObject instanceMethodSignatureForSelector:@selector(init)];
    }
    
    //这里设计很巧妙,一般人是这样调用的 [anInvocation invokeWithTarget:self.obj];
    //但是作者只修改了返回值为null,并不调用方法。因为真正的方法调用根本不在这里,他在- (id)forwardingTargetForSelector:(SEL)selector这里就是实现了调用。
    //所以郭老师复写两个方法,目的只有一个:避免抛出异常:unrecognized selector sent to instance
    - (void)forwardInvocation:(NSInvocation *)invocation {
        void *null = NULL;
        [invocation setReturnValue:&null];
    }
    
    #pragma mark - <NSObject>
    // 重写NSObject.h定义的方法
    //这个方法特殊,它返回NO,就都不会执行了
    - (BOOL)respondsToSelector:(SEL)aSelector {
        return [_target respondsToSelector:aSelector];
    }
    
    - (BOOL)isEqual:(id)object {
        return [_target isEqual:object];
    }
    
    - (NSUInteger)hash {
        return [_target hash];
    }
    
    - (Class)superclass {
        return [_target superclass];
    }
    
    - (Class)class {
        return [_target class];
    }
    
    - (BOOL)isKindOfClass:(Class)aClass {
        return [_target isKindOfClass:aClass];
    }
    
    - (BOOL)isMemberOfClass:(Class)aClass {
        return [_target isMemberOfClass:aClass];
    }
    
    - (BOOL)conformsToProtocol:(Protocol *)aProtocol {
        return [_target conformsToProtocol:aProtocol];
    }
    
    - (BOOL)isProxy {
        return YES;
    }
    
    - (NSString *)description {
        return [_target description];
    }
    
    - (NSString *)debugDescription {
        return [_target debugDescription];
    }
    
    @end
    

    参考文章:1.关于NSProxy的理解
    2.利用消息转发机制屏蔽unrecognized selector sent to instance

    相关文章

      网友评论

          本文标题:2019-03-07 对WKWebView中的循环引用的理解

          本文链接:https://www.haomeiwen.com/subject/qobcpqtx.html