首页 > 代码库 > 【边做项目边学Android】异常处理:android.os.NetworkOnMainThreadException--多线程问题

【边做项目边学Android】异常处理:android.os.NetworkOnMainThreadException--多线程问题

一切搞定。以为高枕无忧了,结果还是有问题!

log開始报错了,获取更新信息异常。。!debug一下。发现Exception:android.os.NetworkOnMainThreadException

这个异常大概意思是在主线程訪问网络时出的异常。 Android在4.0之前的版本号 支持在主线程中訪问网络。可是在4.0以后对这部分程序进行了优化,也就是说訪问网络的代码不能写在主线程中了。

查看网上的解决方法。在Android中实现异步任务机制有两种方式,Handler和AsyncTask。

Handler模式须要为每个任务创建一个新的线程,任务完毕后通过Handler实例向UI线程发送消息,完毕界面的更新,这样的方式对于整个过程的控制比較精细,但也是有缺点的。比如代码相对臃肿,在多个任务同一时候运行时,不易对线程进行精确的控制。

为了简化操作。Android1.5提供了工具类android.os.AsyncTask。它使创建异步任务变得更加简单,不再须要编写任务线程和Handler实例就可以完毕同样的任务。

这里我们採用AsyncTask的处理方式。

先来看看AsyncTask的定义:

public abstract class AsyncTask<Params, Progress, Result> {

三种泛型类型分别代表“启动任务运行的输入參数”、“后台任务运行的进度”、“后台计算结果的类型”。

在特定场合下。并非全部类型都被使用。假设没有被使用。能够用java.lang.Void类型取代。

AsyncTask的运行分为四个步骤

每一步都相应一个回调方法,这些方法不应该由应用程序调用。开发人员须要做的就是实现这些方法。
1) 子类化AsyncTask
2) 实现AsyncTask中定义的以下一个或几个方法
         onPreExecute(), 该方法将在运行实际的后台操作前被UI thread调用。

能够在该方法中做一些准备工作,如在界面上显示一个进度条。
        doInBackground(Params...), 将在onPreExecute 方法运行后立即运行。该方法运行在后台线程中。这里将主要负责运行那些非常耗时的后台计算工作。能够调用 publishProgress方法来更新实时的任务进度。

该方法是抽象方法,子类必须实现。
        onProgressUpdate(Progress...),在publishProgress方法被调用后,UI thread将调用这种方法从而在界面上展示任务的进展情况,比如通过一个进度条进行展示。
        onPostExecute(Result), 在doInBackground 运行完毕后,onPostExecute 方法将被UI thread调用。后台的计算结果将通过该方法传递到UI thread.

 

为了正确的使用AsyncTask类。下面是几条必须遵守的准则:

1) Task的实例必须在UI thread中创建
2) execute方法必须在UI thread中调用
3) 不要手动的调用onPreExecute(), onPostExecute(Result),doInBackground(Params...), onProgressUpdate(Progress...)这几个方法
4) 该task仅仅能被运行一次。否则多次调用时将会出现异常
      doInBackground方法和onPostExecute的參数必须相应。这两个參数在AsyncTask声明的泛型參数列表中指定。第一个为 doInBackground接受的參数,第二个为显示运行进度的參数。第第三个为doInBackground返回和onPostExecute传入的參数。

以下是项目中详细的处理方法:

package com.liuhao.mobilesafe.engine;

import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.URL;

import android.content.Context;
import android.os.AsyncTask;

import com.liuhao.mobilesafe.domain.UpdateInfo;

public class UpdateInfoService {

	private Context context; //应用程序环境的上下文信息

	public UpdateInfoService(Context context) {
		this.context = context;
	}

	// 将与网络通信的过程封装在ServiceInBackGround的doInBackground方法中
	private class ServiceInBackGround extends AsyncTask<Integer, Void, UpdateInfo>{

		@Override
		protected UpdateInfo doInBackground(Integer... params) {
			String path = context.getResources().getString(params[0]); //依据urlId获取资源文件里相应的内容
			UpdateInfo info = new UpdateInfo();
			URL url;
			try {
				url = new URL(path);
				HttpURLConnection conn = (HttpURLConnection) url.openConnection();
				conn.setConnectTimeout(20000);
				conn.setRequestMethod("GET");
				
				InputStream is = conn.getInputStream(); //得到url相应的文件流。应该是xml文件流,须要对其进行解析
				
				info = UpdateInfoParser.getUpdateInfo(is); //对xml文件流进行解析,获取到更新信息实体
			} catch (Exception e) {
				e.printStackTrace();
			}
			return info;
		}
		
		@Override
		protected void onPostExecute(UpdateInfo result) {
			super.onPostExecute(result);
		}
		
	}
	
	/**
	 * @param urlId
	 *            server资源路径相应的id
	 * @return 更新信息
	 * @throws Exception
	 */
	public UpdateInfo getUpdateInfo(int urlId) throws Exception {
//		new serviceInBackGround().execute(urlId).get();
		return new ServiceInBackGround().execute(urlId).get();
	}

}


最后须要说明AsyncTask不能全然代替线程,在一些逻辑较为复杂或者须要在后台重复运行的逻辑就可能须要线程来实现了。

异常都解决后。再次执行程序:

点击进入程序,会弹出升级对话框:

技术分享

查看日志:

技术分享

用户点击确定:

技术分享

点击取消:

技术分享


參考:

http://www.cnblogs.com/dawei/archive/2011/04/18/2019903.html

http://daoshud1.iteye.com/blog/1843655

【边做项目边学Android】异常处理:android.os.NetworkOnMainThreadException--多线程问题