Android访问网络有很多种方式,比如WiFi,移动网络(3G等)...
他们是有一个优先级的。
- 当WiFi连接,打开3G网络,3G网络将被自动屏蔽;
- 当3G连接,连接WiFi,3G网络将被屏蔽,启用WiFi;
- 当WiFi连接,然后关闭,如果3G网络可用,将自动切换到3G网络。
这是系统默认的,可以通过API来改变这种优先级。比如:当3G连接,连接WiFi,我们希望仍然使用3G网络,那么
ConnectivityManager connMgr = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
connMgr.setNetworkPreference(ConnectivityManager.TYPE_MOBILE);
当这样设置之后,会自动断开WiFi连接。
那通过代码,怎么知道WiFi是否连接或者移动网络是否连接呢?
//判断是否有网络连接
public boolean isOnline() {
ConnectivityManager connMgr = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connMgr.getActiveNetworkInfo();
//connMgr.setNetworkPreference(ConnectivityManager.TYPE_MOBILE);
return (networkInfo != null && networkInfo.isAvailable());
}
//判断WiFi是否连接
public boolean isWifiConnected() {
ConnectivityManager connMgr = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connMgr.getNetworkInfo(ConnectivityManager.TYPE_WIFI);
return (networkInfo != null && networkInfo.isConnected());
}
//判断移动网络是否连接
public boolean isMobileConnected() {
ConnectivityManager connMgr = (ConnectivityManager)
getSystemService(Context.CONNECTIVITY_SERVICE);
NetworkInfo networkInfo = connMgr.getNetworkInfo(ConnectivityManager.TYPE_MOBILE);
return (networkInfo != null && networkInfo.isConnected());
}
判断WiFi是否连接,还可以通过WifiManager
//判断WiFi是否连接
public boolean isWifiConnected() {
WifiManager wifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
if (wifiManager.isWifiEnabled()) {
WifiInfo wifiInfo = wifiManager.getConnectionInfo();
return wifiInfo != null && SupplicantState.COMPLETED == wifiInfo.getSupplicantState();
}
return false;
}
它还提供了很多操控WiFi的方法,比如连接或断开WiFi连接,打开或关掉WiFi等。
有WiFi连接并不等于连接到Internet。比如:手机连接到无线路由器,而无线路由器没有连接Internet,这种情况可以用代码容易地判断
//判断WiFi连接时是否能够访问Internet
public boolean hasWifiInternetAccess() {
WifiManager mWifiManager = (WifiManager) getSystemService(Context.WIFI_SERVICE);
if(mWifiManager.isWifiEnabled()) {
return mWifiManager.pingSupplicant();
}
return false;
}
上面的方法其实并不能准确的判断当WiFi连接是是否可以访问Internet。例如,当连接到机场的WiFi,它需要打开浏览器,在弹出的页面上输入用户名密码,此时pingSupplicant返回的是TRUE,当这种情况时,只能通过代码访问某个网址,检查返回值来检测是否能够上网。
通常,为了更好的用户体验,比较大的网络请求只在有WiFi的情况下才发出,比如下载地图数据,下载高质量的音乐。因为网络请求是一个比较耗时的操作,所以不能放在主线程(UI线程)进行,了解更多请点击这里。可以使用Android中的AsyncTask发送网络请求。
TextView textView;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_network_test);
textView = (TextView) findViewById(R.id.textView);
NetworkTask networkTask = new NetworkTask();
networkTask.execute();
}
private class NetworkTask extends AsyncTask {
String statusCode = "";
@Override
protected Void doInBackground(Void... voids) {
//耗时操作。在这里是发送网络请求,系统会将本方法里面的代码在新的工作线程里面执行
DefaultHttpClient client = new DefaultHttpClient();
HttpParams params = client.getParams();
int timeoutInMillisecond = 10000;
HttpConnectionParams.setConnectionTimeout(params, timeoutInMillisecond);
HttpConnectionParams.setSoTimeout(params, timeoutInMillisecond);
int retryTimes = 1;
if (retryTimes > 0) {
client.setHttpRequestRetryHandler(new DefaultHttpRequestRetryHandler(retryTimes, true));
}
HttpGet get = new HttpGet("http://www.cchen.com");
HttpResponse response = null;
try {
response = client.execute(get);
statusCode = String.valueOf(response.getStatusLine().getStatusCode());
} catch (IOException e) {
statusCode = "-1";
}
return null;
}
@Override
protected void onPostExecute(Void aVoid) {
//更新UI。上面方法执行完毕后,系统会在主线程(UI线程)执行本方法中的代码
textView.append(statusCode + "\n");
}
}
注意,上面的HttpConnectionParams设置了两个Timeout:
- ConnectionTimeOut,建立连接的timeout
- SoTimeout,socket timeout,等待返回数据的timeout
上面代码设置两个timeout都是10秒。那假如手机不能上网,是不是10秒之后才会抛出异常呢?答案是否定的。
- 如果手机不能上网,这里会立即报错。注意,这里指的是上网,不是连接WiFi或者移动网络,因为连接上了WiFi或者移动网络并不等于连接上Internet。
- 如果手机能上网,访问某个网址,连接时间超过10秒,则会报错。例如,上面我给的网址,http://www.cchen.com,这是一个不存在的网址,因此,程序在10秒之后,屏幕显示错误码-1.
还有当我们听着音乐回到家中,手机自动连接到WiFi,这个时候程序就应该播放高质量的音乐了,那我们怎么知道手机的网络连接类型发生了变化呢?Android系统在手机网络类型发生变化的时候,会发送一个广播
ConnectivityManager.CONNECTIVITY_ACTION
所以,我们只需要监听这个广播,再利用上面的代码,就能轻易得知当前网络类型,然后进行相关操作。