博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
iOS - 让WKWebView 支持 NSURLProtocol
阅读量:6246 次
发布时间:2019-06-22

本文共 3233 字,大约阅读时间需要 10 分钟。

 

iOS8以后,苹果推出了新框架Webkit,提供了替换UIWebView的组件WKWebView。各种UIWebView的问题没有了,速度更快了,占用内存少了,一句话,WKWebView是App内部加载网页的最佳选择!我们做开发最关系的是内存问题,基本上网上所有的资料都在说WKWebview的内存占用会更少,但是到底少了多少我这边做了下测试,同样是加载163的首页

 
使用UIWebView的内存
 
 
使用WKWebview的内存

从上图看出内存大概能优化百分之八十左右,而且从网页的滑动上也确实有所改善。这么明显的性能提升但是苹果并没有完全放弃UIWebView也一定有他的道理,就拿本文要讲的NSURLProtocol拦截请求来说,WKWebview的兼容并不UIWebView好,还需要开发者做一些操作。

WebKit源码分析

由于WKWebview是基于webkit内核来做的,所以我们在使用的时候需要导入一个这样的东西。

#import <WebKit/WebKit.h>
通过这个我们可以猜到WKWebview中所有的请求以及一些逻辑肯定走的都是webkit里面的东西,所以他对于网页的加载之之类的操作也不会走系统本省的URL Loading System,这么说来他的请求不能被NSURLProtocol拦截也是理所当然的了。不过WKWebview是否真的和NSURLProtocol一点关系都没有还需要去研究,幸好webkit是开源的,github上很容易找到(大小大概是1G多点的zip,花了我将近一天时间来看)。拉下代码直接搜索NSURLProtocol,看看有没有有关的信息

 
搜索结果
看来的确是有和NSURLProtocol有关系,后面通过断点的调用栈中也找到了
+ [NSURLProtocol canInitWithRequest:]
这样的字样,再通过网上查一些资料也证实了我的猜想,其实WKWebview在一开始时候是会调用到NSURLProtocol中的入口方法canInitWithRequest的,但是就没有然后了,也就是说WKWebview是和NSURLProtocol有一定关联,只是在NSURLProtocol的入口处返回NO所以导致NSURLProtocol不接管WKWebview的请求。我们点进webkit源码中的CustomProtocol可以看到,整体的结构我们都差不多,但是我注意到每个CustomProtocol的入口函数都有这样一个判断:
 
入口函数1
 
入口函数2
(粉色的可以暂时认定为是它内部的一个custom字符串)通过这个可以猜想,WKWebview并不是不走NSURLProtocol,而是需要满足他的一个规则,他才会在入口函数这里返回YES来给你放行,这个规则便是你所请求的URL的Scheme要和它内部配置的CustomScheme相同。不过这里有一个疑问,苹果在使用webkit时候为什么会把http/https这样大众化的scheme过滤掉,看来他是不建议开发者来使用NSURLProtocol。接下来我们来看这个CustomScheme,既然苹果内部规定好的那么一定能通过某种方式来注册一个自己的scheme,实在不行就hook嘛。通过翻他的源码发现最终都指向一句代码
[WKBrowsingContextController registerSchemeForCustomProtocol:testScheme];

方法实现为

+ (void)registerSchemeForCustomProtocol:(NSString *)scheme{WebProcessPool::registerGlobalURLSchemeAsHavingCustomProtocolHandlers(scheme);}void WebProcessPool::registerGlobalURLSchemeAsHavingCustomProtocolHandlers(const String& urlScheme){    if (!urlScheme)        return;    globalURLSchemesWithCustomProtocolHandlers().add(urlScheme);    for (auto* processPool : allProcessPools())        processPool->registerSchemeForCustomProtocol(urlScheme);}

  

  通过方法名字可以看出这个就是那个向webkit注册CustomScheme的方法,只要我们在注册完我们自己的CustomProtocol之后在调用该方法应该就可以了。通过他的源码也进一步印证了我的猜想(他也是这么写的)

 
webkit源码
具体实施

找到了方法就要去实施,不过因为registerSchemeForCustomProtocol是WKBrowsingContextController的类方法,所以只能用WKBrowsingContextController去调用,但是在webkit的头文件发现WKBrowsingContextController并没有开放出来,所以我们采用NSClassFromString和NSSelectorFromString方法来拿到类和对应的方法,整体代码如下

 
//注册自己的protocol    [NSURLProtocol registerClass:[CustomProtocol class]];    //创建WKWebview    WKWebViewConfiguration * config = [[WKWebViewConfiguration alloc] init];    WKWebView * wkWebView = [[WKWebView alloc] initWithFrame:CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, [UIScreen mainScreen].bounds.size.height) configuration:config];    [wkWebView loadRequest:webViewReq];    [self.view addSubview:wkWebView];    //注册scheme    Class cls = NSClassFromString(@"WKBrowsingContextController");    SEL sel = NSSelectorFromString(@"registerSchemeForCustomProtocol:");    if ([cls respondsToSelector:sel]) {        // 通过http和https的请求,同理可通过其他的Scheme 但是要满足ULR Loading System        [cls performSelector:sel withObject:@"http"];        [cls performSelector:sel withObject:@"https"];    }

 

值得注意
  • 关于私有API

因为WKBrowsingContextController和registerSchemeForCustomProtocol应该是私有的所以使用时候需要对字符串做下处理,用加密的方式或者其他就可以了,实测可以过审核的。

 

第三方的库

 

转载于:https://www.cnblogs.com/junhuawang/p/8466128.html

你可能感兴趣的文章
我的友情链接
查看>>
不常用sql语法
查看>>
ftp命令
查看>>
Java内存分配全面浅析
查看>>
32、OSPF在帧中继中不同网络类型配置总结
查看>>
git 之CAfile问题
查看>>
EBS exit_form()
查看>>
手动安装Jenkins插件
查看>>
共享主机
查看>>
阿里巴巴Dubbo实现的源码分析
查看>>
exe4j,
查看>>
Mysql查看执行计划
查看>>
SCDPM 2010系列之一——安装
查看>>
cocos2dx学习
查看>>
http加密访问应用
查看>>
vlayout
查看>>
必读 | 什么时候开始准备2019年下半年的考试?
查看>>
JDK安装说明
查看>>
iftop-流量监控安装(脚本)
查看>>
Windows Server2008通过命令行方式添加防火墙规则
查看>>