38 热点问题答疑(四) 你好,我是戴铭。今天这篇答疑文章,我要针对近期留言中的热点问题,进行一次集中解答。

目前,我们专栏已经更新完了基础篇、应用开发篇和原理篇3大模块的内容。其中,原理篇的内容,因为涉及到的都是底层原理,比如系统内核XNU、AOP、内存管理和编译等,学习起来会很辛苦。但所谓良药苦口,你只有搞明白了这些最最底层的原理,才可以帮你抓住开发知识的规律,达到融会贯通的效果,进而提升自己造轮子、解决问题的能力。

也正因为这些底层知识比较难啃,需要细细琢磨,所以在这期答疑文章中,我并没有展开这个模块的内容。如果你对这个模块的文章有哪里不理解,或者觉得哪里有问题的话,可以在评论区留下你的观点,我会挑选合适的时机,给你答复。

接下来,我们就看看今天这篇文章要展开讨论的问题吧。

关于监控卡顿

@凡在第13篇文章《如何利用 RunLoop 原理去监控卡顿?》后问道: 大多数的卡顿监控,都是在主线程上做的。音视频播放以及直播的卡顿,能否使用这种方式来监控呢?另外,我们公司对接的直播都是第三方的库和知识平台,我应该如何把这种监控放到客户端来做呢?

针对这个同学的问题,我想说的是,只有在主线程上卡了,用户才会感知到,而监控卡顿主要就是要监控什么时候会卡。只要我们在发生卡顿的时刻,想办法去收集卡顿信息,就能够定位到问题,找出具体是由谁引起的卡顿。

比如,@凡同学提到的音视频播放卡顿问题,监控到发生卡顿的时刻,通过获取当时方法调用堆栈的方式,就能够确定出具体是哪个方法在调用,从而找到发生卡顿问题的原因。

当然,有些时候只通过各个线程中的方法调用栈来分析问题,可能信息还不太够,这时你还可以捕获各线程卡顿时的 CPU 使用率,进而发现哪个方法占用资源过高。同时,你还能够通过业务场景和环境数据埋点信息,综合分析发生卡顿时,业务场景以及数据是否出现了异常。

关于SMLogger的实现

@梁华建在第9篇文章《无侵入的埋点方案如何实现?》后留言,想要知道SMLogger是如何实现的。

SMLogger,是我对日志记录的一个封装。我在第9篇文章中使用 SMLogger 的方式,是这样的: [[[[SMLogger create] message:[NSString stringWithFormat:@”%@ Appear”,NSStringFromClass([self class])]] classify:ProjectClassifyOperation] save];

可以看出,我把SMLogger 的接口设计成了链式调用的方式。这样的接口接收外部数据后,能够更加灵活地进行组合。

对于日志记录来说,可以设置默认的日志分类和日志级别,简单记录日志描述就只需要一个日志描述数据。

当使用者需要日志库记录一个对象时,就需要增加一个新的接口来支持记录对象。接下来,就会面对外部输入会进行不同组合的情况,比如日志记录对象、日志描述、日志分类、日志级别这四个数据的不同组合。为了满足这些不同的组合,你设置的接口数量也会增加很多。如果都放到一个统一接口中当作不同参数,那么参数的个数就会非常多,导致接口使用起来非常不方便。比如,你每次只需要设置日志描述这个参数,但是使用了多参数的统一接口后,需要手动去设置其他参数值。

使用链式调用的好处就是可以随意组合。而且,当有新的输入类型加入,要和以前接口组合时,也不需要额外工作。我定义的 SMLogger 的链式接口,如下所示: //初始化 + (SMLogger /)create; //可选设置 - (SMLogger /)object:(id)obj; //object对象记录 - (SMLogger /)message:(NSString /)msg; //描述 - (SMLogger /)classify:(SMProjectClassify)classify; //分类 - (SMLogger /)level:(SMLoggerLevel)level; //级别 //场景记录 - (SMLogger /*)scene:(SceneType)scene; //最后需要执行这个方法进行保存,什么都不设置也会记录文件名,函数名,行数等信息 - (void)save;

可以看出,日志记录对象、日志描述、日志分类、日志级别分别为 object、message、classity、level。当需要在日志记录中增加业务场景数据时,只需要简单增加一个 scene 链式接口,就能够达到组合使用业务场景数据和其他链式接口的目的。

在 SMLogger 中,我还在链式基础上实现了宏的方式,来简化一些常用的日志记录接口调用方式。宏的定义如下: // 宏接口 FOUNDATION_EXPORT void SMLoggerDebugFunc(NSUInteger lineNumber, const char /functionName, SMProjectClassify classify, SMLoggerLevel level, NSString /format, …) NS_FORMAT_FUNCTION(5,6); // debug方式打印日志,不会上报 /#ifdef DEBUG /#define SMLoggerDebug(frmt, …) SMLoggerCustom(SMProjectClassifyNormal,SMLoggerLevelDebug,frmt, /#/#VA_ARGS) /#else /#define SMLoggerDebug(frmt, …) do {} while (0) /#endif // 简单的上报日志 /#define SMLoggerSimple(classify,frmt, …) SMLoggerCustom(classify,SMLoggerLevelInfo,frmt, /#/#VA_ARGS) // 自定义classify和level的日志,可上报 /#define SMLoggerCustom(classify,level,frmt, …) \ do { SMLoggerDebugFunc(LINE,FUNCTION,classify,level,frmt, /#/#VA_ARGS);} while(0)

可以看到,宏定义最终调用的是 SMLoggerDebugFunc 函数,这个函数的实现如下所示:

void SMLoggerDebugFunc(NSUInteger lineNumber, const char /functionName, SMProjectClassify classify, SMLoggerLevel level, NSString /format, …) { va_list args; if (format) { va_start(args, format); // 输出方法名和行号 NSString /*msg = [[NSString alloc] initWithFormat:format arguments:args]; msg = [NSString stringWithFormat:@”[%s:%lu]%@”,functionName,(unsigned long)lineNumber,msg]; // SMLogger 链式调用 [[[[[SMLogger create] message:msg] classify:classify] level:level] save]; va_end(args); } }

通过上面代码可以看到,SMLoggerDebugFunc 在处理完方法名和行号后,最终使用的就是SMLogger 链式调用。

通过宏的定义,日志记录接口调用起来也会简化很多,使用效果如下: // 宏方式使用,会记录具体调用地方的函数名和行数 SMLoggerDebug(@”此处必改:%@ 此处也必改: %@”,arr,dict); //仅调试,不上报 SMLoggerSimple(SMProjectClassifyNormal,@”此处必改:%@ 此处也必改: %@”,arr,dict); //会上报 SMLoggerCustom(SMProjectClassifyNormal,SMLoggerLevelDebug, @”这两个需要上报%@%@”,arr,dict); //level为debug不上报

NSURLProtocol相关

@熊在第28篇文章《怎么应对各种富文本表现需求?》后留言到: WKWebView 对NSURLProtocol的支持不太好,我在网上找到的方法都不适用,连Ajax请求都不好去拦截。

其实,WKWebView 处理资源缓存的思路和 UIWebView 类似,需要创建一个 WKURLSchemeHandler,然后使用 -[WKWebViewConfiguration setURLSchemeHandler:forURLScheme:] 方法注册到 WKWebView 配置里。

WKURLSchemeHandler 实例可以用来处理对应的 URLScheme 加载的资源,使用它的 webView:startURLSchemeTask 方法可以加载特定资源的数据。这样就能够起到和 NSURLProtocol 同样的效果。

关于JSON解析的问题

@大太阳在第26篇文章《如何提高JSON解析的性能?》中留言到: 我现在项目是用Swift语言开发的,绝大部分的JSON解析用的是SwiftyJSON,很少一部分用到了KVC。我想问下,SwiftyJSON的效率怎么样?我怎么才能评测这个效率?市面上比较出名的第三方库,它们的效率排名是什么样的?

其实,市面上的大多数第三方库,在解析 JSON 时用的都是系统自带的 JSONSerialization。因此,从本质上来看,它们的解析效率并无差别,只是在易用性、容错率、缓存效率上有些许差异。

比如,@大太阳提到的 SwiftyJSON 库,初始化方法如下: public init(data: Data, options opt: JSONSerialization.ReadingOptions = []) throws { let object: Any = try JSONSerialization.jsonObject(with: data, options: opt) self.init(jsonObject: object) }

可以看到,SwiftyJSON 库在解析JSON时,使用的是 JSONSerialization。你可以点击这个链接,查看SwiftJSON 的完整代码。

既然 SwiftyJSON 也是使用JSONSerialization 来解析JSON的,那么解析效率就和其他使用JSONSerialization 解析的第三方库相比,没有本质上的差别。

JSON案例相关

@徐秀滨在第23篇文章《如何构造酷炫的物理效果和过场动画效果?》后留言反馈,对通过JSON来控制代码逻辑的能力这块内容,感觉理解起来有些困难。接下来,针对这个问题,我再多说两句,希望能够对你有多帮助。

我在第26篇文章《如何提高JSON解析的性能?》中,举了个更加具体的例子,使用JSON 描述了一段 JavaScript 代码逻辑,你可以先看一下这篇文章的相关内容。

对于开发者来说,App 中的任何逻辑都可以通过代码来描述,而代码又能够转换成抽象语法树结构。JSON 作为一种数据结构的表示,同样可以表示代码的抽象语法树,自然也能够具有控制代码逻辑的能力。

总结

今天这篇答疑文章,我和你分享了监控卡顿、SMLogger、NSURLProtocol、JSON 相关的问题。

监控卡顿的方案实际上是通用的,和具体的场景没有关系。卡只是表现在主线程上,根本原因还是需要分析每个线程。

通过NSURLProtocol 对 WKWebView 支持不好的问题,我们可以看出,苹果公司为了更好地管控 WKWebView 而增加了一层,将资源的加载处理单独提供出来供开发者使用,以满足开发者自定义提速的需求。

最后,JSON 解析效率的提高,还是需要从根本上去解决,封装层解决的是易用性问题,所加缓存也只能解决重复解析的问题。

感谢你的收听,欢迎你在评论区给我留言分享你的观点,也欢迎把它分享给更多的朋友一起阅读。

参考资料

https://learn.lianglianglee.com/%e4%b8%93%e6%a0%8f/iOS%e5%bc%80%e5%8f%91%e9%ab%98%e6%89%8b%e8%af%be/38%20%e7%83%ad%e7%82%b9%e9%97%ae%e9%a2%98%e7%ad%94%e7%96%91%ef%bc%88%e5%9b%9b%ef%bc%89.md