Android6.0之后系统中优先级设置都是根据Score分值来设置优先级,分值0-100,数值越高,越优先。
-
- SIM卡网络 50
- wifi网络 60
- 有线网络 70
-
手机网络设置都有自己的Factory设置类,都继承自NetworkFactory.java
wifi网络设置类:WifiNetworkFactory.java packages/modules/Wifi/service/java/com/android/server/wifi/WifiNetworkFactory.java
有线网络设置类:EthernetNetworkFactory.java frameworks\opt\net\ethernet\java\com\android\server\ethernet\EthernetNetworkFactory.java
移动网络设置类:TelephonyNetworkFactory.java
frameworks\opt\telephony\src\java\com\android\internal\telephony\dataconnection\TelephonyNetworkFactory.java
NetworkFactory的子类都有NETWORK_SCORE常量,表示该网络的分值。
private final static int NETWORK_SCORE = 55; //change score from 70
但是我Android11 的代码修改后发现并不能生效,还是有线网优先。
研究了一下EthernetNetworkFactory.java和ConnectivityService.java发现里面的逻辑有很大的修改。
找到EthernetNetworkFactory.java的getNetworkScore()方法,这里面返回的score才是有线网的有效分值;
这个getNetworkScore()方法是在Android11 新增的。
在该方法返回NETWORK_SCORE值即可。里面很多判断是没啥用的。
如果要wifi优先级高于有线,一定要设置有线网络的分值比wifi小,在后期测试过程中发现在某些情况,wifi的分值会变成20,把有线网络分值设置成15才生效。
adb是可以进行分值查看的,文章最后有描述。
参考:
Framework中的连接管理机制: https://blog.csdn.net/u010961631/article/details/48629601
网络连接评分机制之NetworkFactory: https://blog.csdn.net/u010961631/article/details/48971431
网络连接评分机制之NetworkAgent: https://blog.csdn.net/u010961631/article/details/48971651
Android网络优先级及更改: https://blog.csdn.net/u013686019/article/details/51447129/
网上的代码都比较旧了,只能做思路参考,里面有些方法不一样了。
ConnectivityService.java和WMS、AMS一样,都是在System_server里面启动的;
ConnectivityService.java里面会对NetworkFactory的网络连接子类进行保存和管理。
Android11 ConnectivityService
新增了NetworkProviderInfo是一个内部类,但是以前的NetworkAgentInfo还保留了;
NetworkFactory.java的evalRequest方法是评分的主要判断逻辑
-
- private void evalRequest(NetworkRequestInfo n) {
-
- if (VDBG) {
- log("evalRequest");
- log(" n.requests = " + n.requested);
- log(" n.score = " + n.score);
- log(" mScore = " + mScore);
- log(" request.providerId = " + n.providerId);
- log(" mProvider.id = " + mProvider.getProviderId());
- }
-
- if (shouldNeedNetworkFor(n)) { //通过一些属性值判断是否需要请求网络
- if (DBG) log(" needNetworkFor");
- needNetworkFor(n.request, n.score);
- n.requested = true;
- } else if (shouldReleaseNetworkFor(n)) { //通过一些属性值判断是否需要释放网络
- if (DBG) log(" releaseNetworkFor");
- releaseNetworkFor(n.request);
- n.requested = false;
- } else {
- if (DBG) log(" done");
- }
- }
-
具体情况和释放网络的操作都是在子类中进行实现的。
真正要使用的网络一定要经过needNetworkFor这个方法。
可以多添加有一些日志确定是否执行某个方法。
相关的子类:WifiNetworkFactory、EthernetNetworkFactory
内部类:NetworkRequestInfo
内部类:NetworkProviderInfo
####(3)Handler
Messenger对象的理解,不是Message对象哦
因为NetworkFactory是继承自Handler,
并且很多消息的发送接收都是通过Messenger对象进行的;
NetworkRequest 网络请求对象
NetworkAgent 网络代理对象,在EthernetNetworkFactory中有创建
NetworkAgentInfo 网络代理封装对象,在ConnectivityService中创建
消息的传送都是通过Messenger,ConnectivityService–》NetworkAgentInfo–》NetworkAgent–》具体的Factory
Network 网络对象,被包含在网络代理对象里面
EthernetNetworkFactory.NetworkInterfaceState.start()–>new NetworkAgent
Vpn.agentConnect()–>new NetworkAgent
NetworkAgent.register()–>
ConnectivityManager.registerNetworkAgent–>
ConnectivityService.registerNetworkAgent–>new NetworkAgentInfo
如果要查看当前网络对应的Score,
可以通过adb shell dumpsys connectivity,里面的Current Networks有很多相关的数据信息,前提是要先连接上对应的网络。
adb 查看分析网络情况详解:
https://blog.csdn.net/wenzhi20102321/article/details/122161589