这里只是简单的介绍一下java多线程的知识,然后主要讲解的是安卓如何在子线程中更新UI
当我们需要执行一些耗时操作,比如说发起一条网络请求,考虑到网速等其他原因,服务器未必会立刻
响应我们的请求,如果不将这类操作放在子线程里去执行,就会导致主线程被阻塞住,从而影响用户对
软件的正常使用。
1.继承Thread类
class MyThread extends Thread{
@Override
public void run() {
//处理具体的逻辑
}
}
然后通过new MyThread.start()方法启动
2.实现Runnable接口
Thread的构造函数接受一个Runnable参数,而我们new出的MyThread正是一个实现了Runnable接口
的对象,所以可以直接传入。
class MyThread implements Runnable{
@Override
public void run() {
//处理具体的逻辑
}
}
使用了这种写法,启动线程的方法也需要进行相应的改变
MyThread myThread = new MyThread();
new Thread(myThread).start()
如果不想再专门定义一个类去实现Runnable接口,可以使用匿名类的方式
new Thread(new Runnable(){
@Override
public void run() {
//处理具体的逻辑
}
}).start();
二者耦合性区别:
第一种方式,继承Thread类,线程任务和线程对象绑定在一起,耦合性较高,不便于维护
第二种方式,实现Runnable接口,线程任务在实现Runnable接口的类中,线程对象是Thread对象,线程任务和线程对象相分离,耦合性低,便于维护
线程的生命周期:
(1)生命周期的五种状态
新建(new Thread)
当创建Thread类的一个实例(对象)时,此线程进入新建状态(未被启动)。
例如:Thread t1=new Thread();
就绪(runnable)
线程已经被启动,正在等待被分配给CPU时间片,也就是说此时线程正在就绪队列中排队等候得到CPU资源。例如:t1.start();
运行(running)
线程获得CPU资源正在执行任务(run()方法),此时除非此线程自动放弃CPU资源或者有优先级更高的线程进入,线程将一直运行到结束。
死亡(dead)
当线程执行完毕或被其它线程杀死,线程就进入死亡状态,这时线程不可能再进入就绪状态等待执行。
自然终止:正常运行run()方法后终止
异常终止:调用stop()方法让一个线程终止运行
堵塞(blocked)
由于某种原因导致正在运行的线程让出CPU并暂停自己的执行,即进入堵塞状态。
正在睡眠:用sleep(long t) 方法可使线程进入睡眠方式。一个睡眠着的线程在指定的时间过去可进入就绪状态。
正在等待:调用wait()方法。(调用motify()方法回到就绪状态)
被另一个线程所阻塞:调用suspend()方法。(调用resume()方法恢复)
2.常用方法
void run() 创建该类的子类时必须实现的方法
void start() 开启线程的方法
static void sleep(long t) 释放CPU的执行权,不释放锁
static void sleep(long millis,int nanos)
final void wait()释放CPU的执行权,释放锁
final void notify()
static void yied()可以对当前线程进行临时暂停(让线程将资源释放出来)
在子线程中更新UI
和许多其他的GUI库一样,Android的UI也是线程不安全的。也就是说,如果想要更新应用程序里的UI元素,则必须在主线程中进行,否则就会出现异常。
新建AndroidThreadTest项目
修改activity_main.xml文件
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<Button
android:id="@+id/change_text"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Change Text"/>
<TextView
android:id="@+id/text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="Hello world"
android:textSize="20sp"/>
</LinearLayout>
然后修改MainActivity.java
package com.gougoucompany.clarence.androidthreadtest;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
private TextView text;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
text = (TextView) findViewById(R.id.text);
Button changeText = (Button) findViewById(R.id.change_text);
changeText.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch(v.getId()) {
case R.id.change_text:
new Thread(new Runnable(){
@Override
public void run() {
text.setText("Nice to meet you");
}
}).start();
break;
default:
break;
}
}
}
我们运行会发现程序崩溃了,由此证实了Android不允许在子线程中进行UI操作。
android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original
thread that created a view hierarchy can touch its views.
对于这种情况,Android提供了异步消息处理机制.
修改MainActivity.java
package com.gougoucompany.clarence.androidthreadtest;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;
public class MainActivity extends AppCompatActivity implements View.OnClickListener{
public static final int UPDATE_TEXT = 1;
private TextView text;
//新增一个Handler对象,并重写父类的handleMessage方法
private Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case UPDATE_TEXT:
//在这里可以进行UI操作
text.setText("Nice to meet you");
break;
default:
break;
}
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
text = (TextView) findViewById(R.id.text);
Button changeText = (Button) findViewById(R.id.change_text);
changeText.setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch(v.getId()) {
case R.id.change_text:
new Thread(new Runnable(){
@Override
public void run() {
/*在子线程中没有直接进行UI操作,而是创建了一个android.os.Message对象
并将它的what字段指定为UPDATE_TEXT,然后调用Handler的sendMessage()方法将
这条message发送出去,然后Handler会收到message,然后在handleMessage()方法
中对它进行处理*/
Message message = new Message();
message.what = UPDATE_TEXT;
handler.sendMessage(message); //将Message对象发送出去
}
}).start();
break;
default:
break;
}
}
}
注意:handleMessage()方法中的代码就是在主线程当中运行的,可以放心的进行UI更改操作