在实现一个WIFI和4G共存的需求后,设备连接WIFI路由器30米的距离会断开网络,正常情况可达到90米的距离。经验证发现设备连接wifi的score<50的时候就会断开wifi,切到4g网络,而其他能达到90米距离的机型却是在score<30时才会断开(此时距离90米左右)。由此我研究了网络优先级及wifi断开的流程的逻辑。
NetworkRanker是具体的网络优先级计算代码,最终选出最有网络(即决定是否断开其他网络),WIFI就是在这里被认为不是最优网络而被执行teardownUnneededNetwork断开网络。
代码中USE_POLICY_RANKING默认true,所有getBestNetworkByPolicy确定最优网络,这里面有很多Policy刷选条件,经过Log发现断开WIFI是被POLICY_EXITING策略影响:
partitionInto(candidates, nai -> !nai.getScore().hasPolicy(POLICY_EXITING),
accepted, rejected);
if (accepted.size() == 1) return accepted.get(0);
对比正常和异常场景Log:
正常Log:
12-27 19:52:05.903 1193 3205 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING accepted.size: 2 ,rejected.size: 0 candidates.size: 2
12-27 19:52:05.903 1193 3205 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING accepted.get(i).getScore: Score(50 ; KeepConnected : 0 ; Policies : EVER_VALIDATED&IS_VALIDATED)
12-27 19:52:05.903 1193 3205 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING accepted.get(i).getScore: Score(50 ; KeepConnected : 0 ; Policies : EVER_VALIDATED&IS_UNMETERED&IS_VALIDATED)
异常Log:
12-27 20:02:22.363 1188 3200 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING accepted.size: 1 ,rejected.size: 1 candidates.size: 2
12-27 20:02:22.363 1188 3200 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING accepted.get(i).getScore: Score(50 ; KeepConnected : 0 ; Policies : EVER_VALIDATED&IS_VALIDATED)
12-27 20:02:22.363 1188 3200 D TESTTEST: NetworkRanker getBestNetworkByPolicy EXITING rejected.get(0).getScore: Score(45 ; KeepConnected : 0 ; Policies : TRANSPORT_PRIMARY&EXITING&IS_UNMETERED&EVER_USER_SELECTED&IS_VALIDATED)
所以异常Log中的Wifi网络包含EXITING策略从而将该WIFI至于reject中,自然返回的最优网络即4G Mobile。
啥么情况下才会在WIFI中添加EXITING策略呢?
public void adjustScore(int change) {
final int newLegacyScore = mScore.getLegacyInt() + change;
final NetworkScore.Builder builder = new NetworkScore.Builder()
.setLegacyInt(newLegacyScore);
if (mNetworkCapabilities.hasTransport(TRANSPORT_WIFI) && newLegacyScore < 50) {
builder.setExiting(true);
}
mScore = builder.build();
mNetworkAgent.sendNetworkScore(mScore);
}
找到如上代码中包含newLegacyScore < 50的时候会设置EXITTING的策略。