android的UI操作不是线程安全的,同时也只有主线程才能够操作UI,同时主线程对于UI操作有一定的时间限制(最长5秒)。为了能够做一些比较耗时的操作(比如下载、打开大文件等),android提供了一些列机制。《android基础知识02——线程安全》系列文章就是参考了网上许多网友的文章后,整理出来的一个系列,介绍了主要的方法。分别如下:
android基础知识02——线程安全2:handler、message、runnable
android基础知识02——线程安全3:Message,MessageQueue,Handler,Looper
android基础知识02——线程安全4:HandlerThread
android基础知识02——线程安全5: AsyncTask
在上文Timer的例子中,我们提到线程安全这个问题,下面我们来详细介绍一下。
一、线程安全
首先来看一下线程安全的定义:
线程安全:如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的,或者说:一个类或者程序所提供的接口对于线程来说是原子操作或者多个线程之间的切换不会导致该接口的执行结果存在二义性,也就是说我们不用考虑同步的问题 。
当一个程序第一次启动的时候,Android会启动一个LINUX进程和一个主线程。默认的情况下,所有该程序的组件都将在该进程和线程中运行 。主线程(Main Thread)主要负责处理与UI相关的事件,如:用户的按键事件,用户接触屏幕的事件以及屏幕绘图事 件,并把相关的事件分发到对应的组件进行处理。所以主线程通常又被叫做UI线 程。UI线程才能与Android UI工具包中的组件进行交互,在开发Android应用时必须遵守单线程模型的原则: Android UI操作并不是线程安全的并且这些操作必须在UI线程中执行。
当主线程正在做一些比较耗时的操作的时候,如正从网络上下载一个大图片,或者访问数据库,由于主线程被这些耗时的操作阻塞住,无法及时的响 应用户的事件,从用户的角度看会觉得程序已经死掉。如果程序长时间不响应,用户还可能得重启系统。为了避免这样的情况,Android设 置了一个5秒 的超时时间,一旦用户的事件由于主线程阻塞而超过5秒 钟没有响应,Android会弹出一个应用程序没有响应的对话框。
二、例子
下面将通过一个案例来演示这种情况:
本程序将设计和实现查看指定城市的当天天气情况的功能,
1. 首先,需要选择一个天气查询的 服务接口,目前可供选择的接口很多,诸如YAHOO的 天气API和Google提 供的天气API。 本文将选择GOOGLE 的 天气查询API。 该接口提供了多种查询方式,可以通过指定具体城市的经纬度进行查询,也可以通过城市名称进行查询。
2. 用户在输入框内输入需要查询的 城市名称,然后点击查询按钮
3. 当用户点击查询按钮后,使用已 经内置在Android SDK中的HttpClient API来调用GOOGLE 的 天气查询API, 然后解析返回的指定城市的天气信息,并把该天气信息显示在Title上
主要代码如下:
public class WeatherReport extends Activity implements OnClickListener {
private static final String GOOGLE_API_URL = "http://www.google.com/ig/api?weather=";
private static final String NETWORK_ERROR = "网络异常";
private EditText editText;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
editText = (EditText) findViewById(R.id.weather_city_edit);
Button button = (Button) findViewById(R.id.goQuery);
button.setOnClickListener(this);
}
@Override
public void onClick(View v) {
//获得用户输入的城市名称
String city = editText.getText().toString();
//调用Google 天气API查询指定城市的当日天气 情况
String weather = getWetherByCity(city);
//把天气信息显示在title上
setTitle(weather);
}
public String getWetherByCity(String city) {
HttpClient httpClient = new DefaultHttpClient();
HttpContext localContext = new BasicHttpContext();
HttpGet httpGet = new HttpGet(GOOGLE_API_URL + city);
try {
HttpResponse response = httpClient.execute(httpGet, localContext);
if (response.getStatusLine().getStatusCode() != HttpStatus.SC_OK) {
httpGet.abort();
} else {
HttpEntity httpEntity = response.getEntity();
return parseWeather(httpEntity.getContent());
}
} catch (Exception e) {
Log.e("WeatherReport", "Failed to get weather", e);
} finally {
httpClient.getConnectionManager().shutdown();
}
return NETWORK_ERROR;
}
}
Android的UI是单线程(Single-threaded)的。为了避免拖住GUI,一些较费时的对象应该交给独立的线程去执行。但幕后的线程来执行UI对象,Android就会发出错误讯息 CalledFromWrongThreadException。
像上例一样由主线程来负责执行 该操作是错误的。所以我们需要在onClick方 法中创建一个新的子线程来负责调用GOOGLE API来获得天气数据。刚接触Android的 开发者最容易想到的方式就是如下:
public void onClick(View v) {
//创建一个子线程执行耗时的从网络上获取天气信息的操作
new Thread() {
@Override
public void run() {
//获得用户输入的城市名称
String city = editText.getText().toString();
//调用Google 天气API查询指定城市的当日天气 情况
String weather = getWetherByCity(city);
//把天气信息显示在title上
setTitle(weather);
}
}.start();
}
你会发 现Android会 提示程序由于异常而终止。为什么在其他平台上看起来很简单的代码在Android上运行的时候依然会出错呢?如果你观察LogCat中打印的日志信息就会发现这样的错误日志:
android.view.ViewRoot$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
从错误信息不难看出Android禁 止其他子线程来更新由UI thread创建的试图。本例中显示天气信息的title实际是就是一个由UI thread所创建的TextView,所以参试在一个子线程中去更改TextView的时候就出错了。这显示违背了单线程模型的原则:Android UI操作并不是线程安全的并且这些操作必须在UI线 程中执行