机遇
一直在工作上忙着做不完的需求,很久没有学习到新的知识点了。在某次机缘巧合下,让我有机会学习、接触网络相关的工作,接触我梦寐以求ndk相关的开发。刚开始的时候确实压力不小,因为一直都是从事android相关业务工作,从没有从事过正式的ndk开发。后面边学边做,首先是把C++的全部基础语法看熟(因为大学有c语言基础),然后结合项目的具体代码排查bug,从一小块逐步熟悉代码的功能。
介绍
我们本章主要从简单的配置参数介绍Cronet(因为复杂的我也还讲不通),这里主要涉及到如何给Cronet新增init参数,如果你和我一样对Cronet是什么都一头雾水,我推荐你先看两篇文章:Quic是什么/怎么用、官方介绍如何接入Cronet库。首先要对Cronet这个开源库要有个大概的了解,Cronet 本身支持 HTTP 协议、HTTP/2 协议和QUIC协议。
Quic协议(机器翻译)
一、介绍
- QUIC 是一种建立在 UDP 之上的新的多路复用传输。HTTP/3 旨在利用 QUIC 的功能,包括流之间没有 Head-Of-Line 阻塞。
- QUIC 项目最初是作为 TCP+TLS+HTTP/2 的替代方案,旨在改善用户体验,尤其是页面加载时间。
- 由于 TCP 是在操作系统内核和中间盒中实现的,因此对 TCP 进行广泛的重大更改几乎是不可能的。然而,由于 QUIC 是建立在 UDP 之上的,并且传输功能是加密的,所以它没有这样的限制。
二、特性
基于 TCP+TLS 和 HTTP/2 的 QUIC 和 HTTP/3 的主要特性包括
减少连接建立时间 - 在常见情况下为 0 次往返
改进的拥塞控制反馈
多路复用,没有线头阻塞
连接迁移
传输可扩展性
可选的不可靠交付
注意:👉quic相关介绍文档要翻墙的quic介绍文档
Cronet初始化参数
类相关介绍
- ExperimentalCronetEngine.java,该类采用Builder设计模式,主要负责存储的是应用程序设置给Cronet的参数和构建发起网络连接的重要对象。ExperimentalCronetEngine源码地址
//创建并配置 CronetEngine 的实例
val myBuilder = ExperimentalCronetEngine.Builder(context). enableHttp2().enableXXX();
val cronetEngine: CronetEngine = myBuilder.build()
- NativeCronetEngineBuilderImpl.java继承CronetEngineBuilderImpl,主要作用就是把Builder存储的配置参数往C++那层传递。NativeCronetEngineBuilderImpl源码地址
- url_request_context_config.cc文件里声明的数据结构URLRequestContextConfig主要就是把java层Builder传进去的数据进行存储。详细可看下面流程图
新增参数涉及类
按照流程图步骤自己跟下,可以发现应用新增参数主要涉及类
JAVA层:CronetUrlRequestContext.java、CronetEngineBuilderImpl.java、ExperimentalCronetEngine.java、CronetEngine.java、ICronetEngineBuilder.java
C++层:cronet_url_request_context_adapter.cc(新增JAVA层映射到C++层的新参数解析)、url_request_context_config.cc((新增c++层参数字段)
结果图
总结
- 学习新东西上会遇到各种各样的技术,肯定会遇到自己不懂的,要懂得google多搜下。如果搜不到的问题点,要用自己对知识的理解去跳过卡住的问题点。特别是CronetUrlRequestContextJni这个生成技术的点卡了我好久,因为里面的代码都是混淆后的产物,无法直接跟着调用链路查看c++层代码。但是NDK的开发总离不开JNI_类名_函数名的头文件和实现函数,只要去c++层搜一下,就能找到了一模一样的函数了。
- 编译cronet推荐还是ubuntu来弄比较友善点,一起努力搬砖吧