zoukankan      html  css  js  c++  java
  • 黑魔法(method-swizzling)解决第三方库引发的问题

     

    更新于2019年02月21日

    需求

    最近做一个项目中,有个需求,所有网络请求,都不显示 NetworkActvityIndicator(也就是状态栏里旋转的小圈圈).

    解决过程1:

    全局搜索 NetworkIndicator 关键字, 把所有涉及 NetworkIndicator 的代码去除,比如 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES];

    测试并发现新问题

    所有界面都不再显示NetworkActvityIndicator了,唯独一个播放视频的界面依然显示。

    猜想: 第三方库引发的问题

    无论是哪些第三方库,正常情况都会通过 setNetworkActivityIndicatorVisible 来 显示状态栏小圈圈。

    验证过程1

    通过继承 UIApplication 来重写了 setNetworkActivityIndicatorVisible 方法。(如何继承UIApplication,请看这里)并把断点打在这个方法体内。

    测试了正常调用 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 是会触发断点的。但是唯独那个视频界面,没有触发该断点的情况下,正常显示小圈圈。

    验证过程2

    通过 KVO 监听 UIApplication 的 networkActivityIndicatorVisible 属性,结果还是和 验证过程1 的情况一样。
    正常调用 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 是会触发监听, 唯独那个视频界面,没有触发监听的情况下,正常显示小圈圈。

    所以, 视频界面里显示的小圈圈,肯定不是通过常规调用 setNetworkActivityIndicatorVisible 方法显示出来的。

    更新猜想: 第三方库引发的问题,并且不是通过常规方法调用

    验证过程3

    显示小圈圈的情况下,分析了该界面的视图层级,发现在 statusBar 上,有 类型为UIActivityIndicatorView的视图存在(并且怪异的存在了两个)。

    那正常情况下,通过 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 显示小圈圈时,视图层级是如何的呢? 通过分析验证, 也是一样的。

    层级都是 UIStatusBarView -> UIStatusBarForegroundView -> UIStatusBarActivityItemView -> UIActivityIndicatorView

    想到解决方案:

    既然小圈圈都是 UIActivityIndicatorView 类型的视图,而 UIActivityIndicatorView 开始动画常规都是调用 startAnimation 方法。
    那何不使用黑魔法(method swizzling)来重写它的 startAnimation 方法,
    判断它的superView是否为 “UIStatusBarActivityItemView”类型,如果是,则直接跳出。否则,执行原有的 startAnimation方法。

    Talk is cheap. Show me the code.

    以下是 .m 文件的代码

    @implementation UIActivityIndicatorView (HideNetworkActivityIndicator)
    
    + (void)load {
    
        static dispatch_once_t onceToken;
        dispatch_once(&onceToken, ^{
            Class class = [self class];
    
            SEL originalSelector = @selector(startAnimating);
            SEL swizzledSelector = @selector(xxx_startAnimating);
    
            Method originalMethod = class_getInstanceMethod(class, originalSelector);
            Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
    
            // When swizzling a class method, use the following:
            // Class class = object_getClass((id)self);
            // ...
            // Method originalMethod = class_getClassMethod(class, originalSelector);
            // Method swizzledMethod = class_getClassMethod(class, swizzledSelector);
    
            BOOL didAddMethod =
            class_addMethod(class,
                            originalSelector,
                            method_getImplementation(swizzledMethod),
                            method_getTypeEncoding(swizzledMethod));
    
            if (didAddMethod) {
                class_replaceMethod(class,
                                    swizzledSelector,
                                    method_getImplementation(originalMethod),
                                    method_getTypeEncoding(originalMethod));
            } else {
                method_exchangeImplementations(originalMethod, swizzledMethod);
            }
        });
    }
    
    #pragma mark - Method Swizzling
    
    - (void)xxx_startAnimating{
    
        if (self.superview != nil && [NSStringFromClass([self.superview class]) isEqualToString: @"UIStatusBarActivityItemView"]) {
            NSLog(@"黑魔法禁止状态栏的loading显示: %@", self);
        } else {
            [self xxx_startAnimating];
        }
    
    }
    
    
    @end
    

    成功了!!!

    (在xxx_startAnimation方法体内打断点,程序进入视频播放界面,触发断点,看调用栈,果然是第三方库引发的问题。)

    参考资料:https://nshipster.cn/method-swizzling/

  • 相关阅读:
    webstrom的热更新没效果
    按钮文字有间距并居中
    兄弟选择器(+ 和 ~)
    ES6精简要点
    自动类型转换之运算符重载
    自动类型转换之构造函数
    运算符重载(三)
    运算符重载(二)
    运算符重载(一)
    友元
  • 原文地址:https://www.cnblogs.com/ZJT7098/p/hei-mo-fa-methodswizzling-jie-jue-di-san-fang-ku-y.html
Copyright © 2011-2022 走看看