我们知道,当手机上显示网络连接成功时,并不一定可以真的上网。常见的情况是,连上路由后需要进行跳转登录,或者干脆路由连接到网络的端口坏掉。因此在进行网络通讯前,可能需要确认网络是否真正联通。
经典方法
一种常用的方法,也是沿用pc机上查看网络是否联通的方法,就是使用ping命令。要注意ping在windows上和linux上的参数意义是不一样的。使用-c来确定ping的次数。adb shell进入android环境,输入命令,效果如下。这里www.a.shifen.com是百度为了防止攻击而设置的壳地址,可以忽略。
那么现在要做的就是在android代码里使用ping命令,代码如下:
Runtime runtime = Runtime.getRuntime();
try {
Process p = runtime.exec("ping -c 3 www.baidu.com");
int ret = p.waitFor();
Log.i("Avalible", "Process:"+ret);
} catch (Exception e) {
e.printStackTrace();
}
然后我们来看下不同网络状态下代码的执行效果:
- 连通的移动数据网络下:
- 在需要网页认证的wifi下:
- 在wifi打开但没有网络连接,数据也不可用的状态下:
- 在不可用wifi下:
- 在可用wifi下:
基本上只要判断是否是0即可,若是0则网络真正可用。
Android 6.0以上可用方法
在API 21增加了一个类NetworkCapabilities,其中有很多对于网络性能的描述,可以通过ConnectivityManager获得描述当前网络的NetworkCapabilities。而在API23中,增加了一个描述NET_CAPABILITY_VALIDATED。文档中对其描述如下:
Indicates that connectivity on this network was successfully validated. For example, for a network with NET_CAPABILITY_INTERNET, it means that Internet connectivity was successfully detected.
实验表明,当NetworkCapabilities的描述中有VALIDATED这个描述时,此网络是真正可用的。使用如下代码获得当前网络的capabilities:
NetworkCapabilities networkCapabilities = mConnectivityManager.getNetworkCapabilities(mConnectivityManager.getActiveNetwork());
Log.i("Avalible", "NetworkCapalbilities:"+networkCapabilities.toString());
可以看到代码非常简单,将对于NetworkCapabilities的描述打印出来,不同网络状态下结果如下:
- 在可联通的wifi下,可看到出现了VALIDATED的标记:
- 在需要认证的wifi下,可以看到相应区域没有VALIDATED的标记:
- 在不可上网的wifi下,可以看到也没有VALIDATED的标记:
- 在移动数据下,可以看到VALIDATED又出现了:
因此可以通过判断这个标记是否存在来判断网络的连通性。通过如下调用来判断标记存在与否:
networkCapabilities.hasCapability(NetworkCapabilities.NET_CAPABILITY_VALIDATED);
可惜只能6.0以上使用,目前还是又很多6.0下的手机在使用的,因此经典方法还是很重要。
http协议
其实查看网络连接是否成功的本质,就是实际去连下,看看能不能连到。所以理论上任何网络通讯协议都可以来处理这件事。但实际上,开发中大多使用经典方法是有道理的。因为ping命令本身就是为了查看网络连通状况而存在。而使用http协议的话,不但数据包会较大、使用较复杂,很多情况也没法处理。下面来简单地用Get去访问url看看会发生什么。代码如下:
HttpURLConnection httpConn = null;
InputStream inputStream = null;
int respondcode = -1;
try{
URL url = new URL("http://www.baidu.com");
httpConn = (HttpURLConnection)url.openConnection();
httpConn.setRequestMethod("GET");
respondcode = httpConn.getResponseCode();
Log.i("Avalible", "HttpURLConnection1:"+respondcode+" "+httpConn.getHeaderField("Location"));
url = new URL(httpConn.getHeaderField("Location"));
httpConn = (HttpURLConnection)url.openConnection();
httpConn.setUseCaches(false);
httpConn.setRequestMethod("GET");
respondcode = httpConn.getResponseCode();
Log.i("Avalible", "HttpURLConnection2:"+respondcode+" "+httpConn.getHeaderField("Location"));
}catch(Exception e){
e.printStackTrace();
}
由于可能会发生重定向,因此进行两次http请求,并且将重定向的url打印出来查看。不同网络下结果如下:
- 移动数据流量。首先进行了一次跳转,第二次返回200成功:
- 可连通wifi下,完全一样:
- 需要认证wifi下,返回200,但实际上并没有确认可连通www.baidu.com,只是连接到了需要登录的地址。因此实际上是没法通过返回值来判断究竟能否上网的:
- 而在无法上网的情况下,是直接报错的:
因此相对来说,使用http协议,处理的过程更加复杂,返回结果不够明确,相对不适合用来判断网络是否连通。