简介:
Thread : Thread类为底层操作系统的线程体系架构提供一套统一接口。单个操作系统线程和一个Thread对象关联。
Runnable: Runable接口为关联Thread对象的线程提供执行的代码。这些代码放在Runable的void run()方法中,这个方法虽然不接受任何参数且没有返回值,但是有可能抛出异常。
创建Thread和Runnable对象
线程都是通过合适的Thread和Runbale对象进入应用程序。Thread类声明了几个构造方法来初始化Thread对象。其中有几个构造方法需要接受Runnable对象作为参数。
Runnable对象创建:
Runnable r = new Runnable() {
@Override
public void run() {
System.out.println("创建Runnable对象");
}
};
Java1.8之后创建方式:
Runnable r = () -> System.out.println("创建Runnable对象")
创建Runnable对象之后,可以它传递到Thread类接收Runnable作为参数的构造函数中。
Thread t = new Thread(r)
少数的构造函数不会接收Runnable作为参数,例如 Thread()构造方法就不会接收它来初始化线程。你必须继承Thread类而重写它的run()方法(Thread类实现了Runnable接口),并提供方法。如下:
public class MyThread extends Thread {
@Override
public void run() {
// TODO Auto-generated method stub
super.run();
}
}
获取和设置线程的状态
一个Thread对象关联着一条线程的状态,这个状态由线程名称、线程存活标识、线程执行状态等、线程优先级、是否是守护线程等
1 获取和设置线程的名称
设置名称: Thread t1 = new Thread(r,"thread t1"); Or t1.setName("xx")
获取名称: t1.getName();
2 获取线程的存活状态
通过调用Thread的Boolean isAlive()方法判断一条线程的存活状态。当前线程是活的,该方法返回true,反之返回false
一条线程的寿命仅仅起始于它真正在start()方法中被启动起来,而结束于它刚刚离开run()方法,此时线程死亡。
Thread t = new Thread(r);
System.out.println(t.isAlive());
3 获取一条线程的执行状态
线程的执行状态有Thread.State枚举常量表示:
NEW: 表示该线程还没有开始执行
RUNNABLE:该线程正在JVM中执行
BLOCKED: 该状态下的线程被阻塞并等待一个监听锁
WAITING: 改状态下线程无限期地等待另一个线程执行特定的操作
TIMED_WAITING: 该状态下线程在特定时间内等待另一个线程执行特定的操作。
TERMINAED: 该线程已经退出操作
获取线程的执行状态:
Thread t = new Thread(r);
System.out.println(t.getState());
4 获取和设置线程的优先级
当计算机有足够的处理器或者处理内核,操作系统就会为每个处理器或核心分配单独的线程,这些线程可以同时执行,一旦计算机没有足够的处理器或者内核的时候,多条线程只能轮转着使用共享的处理器或者核心了。
Thread通过返回当前优先级的int getPriority()方法以及设置优先级的void setPriority(int priority)方法来实现优先级操作。传递给优先级的值介于Thread.MIN_PRIORITY和Thread.MAX_PRIORITY之间。而Thread.NORMAL_PRIORITY则确定了默认的优先级。
Thread t = new Thread(t);
System.out.println(t.getPriority());
t.setPriority(Thread.MAX_PRIORITY);
警告:使用setPriority()会影响应用程序跨系统操作的可移植性,因为不同的调度器会采用不同的方式处理优先级,举个例子:一个操作系统调度器可能会推迟低优先级线程执行,直到高优先级线程执行完毕,这会导致无限延迟。而当无限期地等待执行,低优先级线程会“饿死”,这会严重损耗应用成的性能。而其他操作系统的调度器可能不会无限期地延迟低优先级的线程,从而改善应用程序的性能。
5 获取和设置线程的守护线程状态
Java将线程分为 守护线程和非守护线程。一条守护线程扮演非守护线程辅助者的角色,并且会在应用程序最后一条非守护线程消失之后自动死亡,因此程序才能终止。
可以通过Thread的Boolean isDaemon()方法来判断线程是否是守护线程,如果是 则返回TRUE,如果不是则返回false
Thread t = new Thread(r);
System.out.println(t.isDaemon());
可以通过Thread的void setDaemon(boolean isDaemon)方法并传入true作为参数,如下:
Thread t = new Thread(r);
t.setDaemon(true);
注意:当非守护默认主线程终止后,应用程序还会等到所以后的非守护线程终止之后才会终止。如果后台的线程本身就是守护线程,那么当默认的主线程终止时,应用程序会立刻终止。
启动线程
在创建一个Thread对象或者其子类对象之后,你可以通过Thread的void start()方法启动与之对象关联的线程。如果之前应启动切处于运行状态,又或者线程死亡 这个方法会抛出状态非法异常! java.lang.IllegalThreadStateException
Thread t =new Thread(r);
r.start();
调用start()方法会在运行时创建底层线程,同时调用run()方法中的指令(start()方法并不会等待所有这些任务都完成才返回)。当run()方法 执行完毕,线程就会被销毁。调用start()方法的Thread对象不再可用。