创建线程
在 Java 程序中创建线程有几种方法每个 Java 程序至少包含一个线程主线程其它线程都是通过 Thread 构造器或实例化继承类 Thread 的类来创建的
Java 线程可以通过直接实例化 Thread 对象或实例化继承 Thread 的对象来创建其它线程在线程基础中的示例(其中我们在十秒钟之内计算尽量多的素数)中我们通过实例化 CalculatePrimes 类型的对象(它继承了 Thread)创建了一个线程
当我们讨论 Java 程序中的线程时也许会提到两个相关实体完成工作的实际线程或代表线程的 Thread 对象正在运行的线程通常是由操作系统创建的Thread 对象是由 Java VM 创建的作为控制相关线程的一种方式
创建线程和启动线程并不相同
在一个线程对新线程的 Thread 对象调用 start() 方法之前这个新线程并没有真正开始执行Thread 对象在其线程真正启动之前就已经存在了而且其线程退出之后仍然存在这可以让您控制或获取关于已创建的线程的信息即使线程还没有启动或已经完成了
通常在构造器中通过 start() 启动线程并不是好主意这样做会把部分构造的对象暴露给新的线程如果对象拥有一个线程那么它应该提供一个启动该线程的 start() 或 init() 方法而不是从构造器中启动它
结束线程
线程会以以下三种方式之一结束
线程到达其 run() 方法的末尾
线程抛出一个未捕获到的 Exception 或 Error
另一个线程调用一个弃用的 stop() 方法弃用是指这些方法仍然存在但是您不应该在新代码中使用它们并且应该尽量从现有代码中除去它们
当 Java 程序中的所有线程都完成时程序就退出了
加入线程
Thread API 包含了等待另一个线程完成的方法join() 方法当调用 Threadjoin() 时调用线程将阻塞直到目标线程完成为止
Threadjoin() 通常由使用线程的程序使用以将大问题划分成许多小问题每个小问题分配一个线程本章结尾处的示例创建了十个线程启动它们然后使用 Threadjoin() 等待它们全部完成
调度
除了何时使用 Threadjoin() 和 Objectwait() 外线程调度和执行的计时是不确定的如果两个线程同时运行而且都不等待您必须假设在任何两个指令之间其它线程都可以运行并修改程序变量如果线程要访问其它线程可以看见的变量如从静态字段(全局变量)直接或间接引用的数据则必须使用同步以确保数据一致性
在以下的简单示例中我们将创建并启动两个线程每个线程都打印两行到 Systemout
public class TwoThreads {
public static class Thread extends Thread {
public void run() {
Systemoutprintln(A);
Systemoutprintln(B);
}
}
public static class Thread extends Thread {
public void run() {
Systemoutprintln();
Systemoutprintln();
}
}
public static void main(String[] args) {
new Thread()start();
new Thread()start();
}
}
我们并不知道这些行按什么顺序执行只知道在之前打印以及A在B之前打印输出可能是以下结果中的任何一种
A B
A B
A B
A B
A B
A B
不仅不同机器之间的结果可能不同而且在同一机器上多次运行同一程序也可能生成不同结果永远不要假设一个线程会在另一个线程之前执行某些操作除非您已经使用了同步以强制一个特定的执行顺序
休眠
Thread API 包含了一个 sleep() 方法它将使当前线程进入等待状态直到过了一段指定时间或者直到另一个线程对当前线程的 Thread 对象调用了 Threadinterrupt()从而中断了线程当过了指定时间后线程又将变成可运行的并且回到调度程序的可运行线程队列中
如果线程是由对 Threadinterrupt() 的调用而中断的那么休眠的线程会抛出 InterruptedException这样线程就知道它是由中断唤醒的就不必查看计时器是否过期
Threadyield() 方法就象 Threadsleep() 一样但它并不引起休眠而只是暂停当前线程片刻这样其它线程就可以运行了在大多数实现中当较高优先级的线程调用 Threadyield() 时较低优先级的线程就不会运行
CalculatePrimes 示例使用了一个后台线程计算素数然后休眠十秒钟当计时器过期后它就会设置一个标志表示已经过了十秒
守护程序线程
我们提到过当 Java 程序的所有线程都完成时该程序就退出但这并不完全正确隐藏的系统线程如垃圾收集线程和由 JVM 创建的其它线程会怎么样?我们没有办法停止这些线程如果那些线程正在运行那么 Java 程序怎么退出呢?
这些系统线程称作守护程序线程Java 程序实际上是在它的所有非守护程序线程完成后退出的
任何线程都可以变成守护程序线程可以通过调用 ThreadsetDaemon() 方法来指明某个线程是守护程序线程您也许想要使用守护程序线程作为在程序中创建的后台线程如计时器线程或其它延迟的事件线程只有当其它非守护程序线程正在运行时这些线程才有用
示例用多个线程分解大任务
在这个示例中TenThreads 显示了一个创建了十个线程的程序每个线程都执行一部分工作该程序等待所有线程全部完成然后收集结果
/**
* Creates ten threads to search for the maximum value of a large matrix
* Each thread searches one portion of the matrix
*/
public class TenThreads {
private static class WorkerThread extends Thread {
int max = IntegerMIN_VALUE;
int[] ourArray;
public WorkerThread(int[] ourArray) {
thisourArray = ourArray;
}
// Find the maximum value in our particular piece of the array
public void run() {
for (int i = ; i < ourArray.length; i++)
max = Math.max(max, ourArray[i]);
}
public int getMax() {
return max;
}
}
public static void main(String[] args) {
WorkerThread[] threads = new WorkerThread[10];
int[][] bigMatrix = getBigHairyMatrix();
int max = Integer.MIN_VALUE;
// Give each thread a slice of the matrix to work with
for (int i=0; i < 10; i++) {
threads[i] = new WorkerThread(bigMatrix[i]);
threads[i].start();
}
// Wait for each thread to finish
try {
for (int i=0; i < 10; i++) {
threads[i].join();
max = Math.max(max, threads[i].getMax());
}
}
catch (InterruptedException e) {
// fall through
}
System.out.println("Maximum value was " + max);
}
}
9、小结
就象程序一样,线程有生命周期:它们启动、执行,然后完成。tw.wInGwit.cOm一个程序或进程也许包含多个线程,而这些线程看来互相单独地执行。
线程是通过实例化 Thread 对象或实例化继承 Thread 的对象来创建的,但在对新的 Thread 对象调用 start() 方法之前,这个线程并没有开始执行。当线程运行到其 run() 方法的末尾或抛出未经处理的异常时,它们就结束了。
sleep() 方法可以用于等待一段特定时间;而 join() 方法可能用于等到另一个线程完成。