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<Void, Void, Void> {
- 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
所以,我们只需要监听这个广播,再利用上面的代码,就能轻易得知当前网络类型,然后进行相关操作。