前言
在使用Android5.0以上的原生系统过程中,大家会遇到明明自己网络是OK的,但是会在状态栏的移动信号或 WiFi 图标旁显示感叹号(5.0 ~ 7.0.0)或叉号(7.1.1),并且会提示“已连接,但不能接入互联网”的情况…
原因
从Android 5.0开始,系统连接网络或切换网络时,系统模块 NetworkMonitor 会向一特定网址发送数据请求,根据响应的结果从而判断网络的连通性。在原生Android系统中,这一特定网址是Google的网址,不翻墙访问不了,故而会出现感叹号或叉号。
解决方案
既然知晓了原理,那么解决方案也很明确了:
完全关闭网络检查服务(不可取):
1
adb shell settings put global captive_portal_detection_enabled 0
缺点: 完全屏蔽该系统功能带来明显的问题,当你使用公共 Wi-Fi 这种需要使用 portal 验证的网络时,因为网络检查被关闭,系统在访问 portal 验证页面时无法返回正确的值,最终导致无法完成验证和上网。
替换检查网络的网址(可取):
通过ADB命令,将验证服务的网址替换为国内正常可以访问的网址
Android 5.x ~ 6.x 解决方案
1 | adb shell "settings put global captive_portal_server www.v2ex.com" |
Android 7.0.0 ~ 7.1.0 解决方案
1 | adb shell "settings put global captive_portal_server captive.v2ex.co" |
Android 7.1.1 解决方案
1 | adb shell "settings put global captive_portal_https_url https://captive.v2ex.co/generate_204" |
源码
问题解决了,问题产生的根本源头在哪呢,在源码里面 :)
Android 5.x ~ 6.x
1 | private int isCaptivePortal() { |
原来的 mServer
是 clients3.google.com
,我们最终将其换为 www.v2ex.com
.
Android 7.0 ~ 7.1.0
1 | private static String getCaptivePortalServerUrl(Context context, boolean isHttps) { |
Android 7.0 ~ 7.1.0 新增了https的验证请求,所以我们把 server
的值改为支持https的 captive.v2ex.co
.
Android 7.1.1
1 | private static String getCaptivePortalServerHttpsUrl(Context context) { |
DEFAULT_HTTPS_URL
的值由原来的 https://www.google.com/generate_204
替换为 https://captive.v2ex.co/generate_204
.