获取和设置线程信息
Thread类的对象中保存了一些属性信息能够帮助我们来辨别每一个线程,知道它的状态,调整控制其优先级。 这些属性是:
- ID: 每个线程的独特标识。
- Name: 线程的名称。
- Priority: 线程对象的优先级。优先级别在1-10之间,1是最低级,10是最高级。不建议改变它们的优先级,但是你想的话也是可以的。
- Status: 线程的状态。在Java中,线程只能有这6种中的一种状态: new, runnable, blocked, waiting, time waiting, 或 terminated.
在这个指南里,我们将开发一个为10个线程设置名字和优先级的程序,然后展示它们的状态信息直到线程结束。这些线程会计算数字乘法表。
准备
指南中的例子是使用Eclipse IDE 来实现的。如果你使用Eclipse 或者其他的IDE,例如NetBeans, 打开并创建一个新的java项目。
怎么做呢…
按照这些步骤来实现下面的例子:
1. 创建一个类名为 Calculator,这个类一定要实现Runnable接口。
1 |
public class Calculator implements Runnable { |
2. 声明一个名为number的private int为属性,然后实现类的构造函数并初始化其值。
2 |
public Calculator( int number) { |
3. 实现方法run()。此方法是给我们创建的线程执行下达指令的,所以这个方法将计算并且打印数字乘法表。
3 |
for ( int i= 1 ; i<= 10 ; i++){ |
4 |
System.out.printf( "%s: %d * %d = %d\n" ,Thread. currentThread().getName(),number,i,i*number); |
4. 现在, 实现应用的主类。创建一个名为Main的类,并包含 main() 方法.
2 |
public static void main(String[] args) { |
5. 创建一个大小为10的Thread类的数组和一个大小为10的Thread.State数组来保存将要执行的线程和它们的状态。
1 |
Thread threads[]= new Thread[ 10 ]; |
2 |
Thread.State status[]= new Thread.State[ 10 ]; |
6. 创建10个Calculator类的对象,每个初始为不同的数字,然后分别用10个线程来运行它们。把其中5个的优先值设为最高,把另外5个的优先值为最低。
1 |
for ( int i= 0 ; i< 10 ; i++){ |
2 |
threads[i]= new Thread( new Calculator(i)); |
4 |
threads[i].setPriority(Thread.MAX_PRIORITY); |
6 |
threads[i].setPriority(Thread.MIN_PRIORITY); |
8 |
threads[i].setName( "Thread " +i); |
7. 创建一个 PrintWriter对象用于把线程状态的改变写入文档。
1 |
try (FileWriter file = new FileWriter( ".\\data\\log.txt" ); PrintWriter pw = new PrintWriter(file);){ |
8. 把10个线程的状态写入文档。现在,它成为NEW.
1 |
for ( int i= 0 ; i< 10 ; i++){ |
2 |
pw.println( "Main : Status of Thread " +i+ " : " +threads[i].getState()); |
3 |
status[i]=threads[i].getState(); |
9. 开始执行这10个线程.
1 |
for ( int i= 0 ; i< 10 ; i++){ |
10. 直到这10个线程执行结束,我们会一直检查它们的状态。如果发现它的状态改变,就把状态记入文本。
03 |
for ( int i= 0 ; i< 10 ; i++){ |
04 |
if (threads[i].getState()!=status[i]) { |
05 |
writeThreadInfo(pw, threads[i],status[i]); |
06 |
status[i]=threads[i].getState(); |
11 |
for ( int i= 0 ; i< 10 ; i++){ |
12 |
finish=finish && (threads[i].getState()==State.TERMINATED); |
11. 实现一个方法 writeThreadInfo(),这个方法写线程的 ID, name, priority, old status, 和 new status。
1 |
private static void writeThreadInfo(PrintWriter pw, Thread thread, State state) { |
2 |
pw.printf( "Main : Id %d - %s\n" ,thread.getId(),thread.getName()); |
3 |
pw.printf( "Main : Priority: %d\n" ,thread.getPriority()); |
4 |
pw.printf( "Main : Old State: %s\n" ,state); |
5 |
pw.printf( "Main : New State: %s\n" ,thread.getState()); |
6 |
pw.printf( "Main : ************************************\n" ); |
12. 运行这个例子,然后打开 log.txt 文档并查看10个线程的状态变化。
它是如何工作的…
接下来是程序在执行的log.txt文本的一些行的裁图。在这个文本中,可以发现有高优先级的线程们比低优先级的先结束。还可以发现线程状态的演变过程。
程序的控制台显示的是线程计算的乘法表,而log.txt文本记录的是不同线程的状态演变。这样子,可以更好的观察线程的演变过程。
Thread 类有能保存使用线程信息的属性。JVM根据线程的优先级来选择将使用CPU的线程,然后再根据每个线程的情况来实现它们的状态。
如果你没有声明一个线程的名字,那么JVM会自动命名它为:Thread-XX,XX是一个数字。线程的ID或者状态是不可修改的。Thread类没有实现setId()和setStatus()方法来允许修改它们。
更多…
在这个指南中,你学习了如何使用Thread对象来访问线程的属性信息。你也可以实现Runnable接口来访问这些信息。你可以用Thread类的静态方法currentThread()来访问正在运行的Runnable 对象的 Thread对象。
你必须知道 setPriority() 方法会抛出 IllegalArgumentException 异常,如果你设置的优先级不是在1-10之间。
参见