1.新建线程
2.终止线程
使用了一个标记
3.线程中断
线程中断比终止线程更强大
4.等待wait()和通知notify()
wait()
- 当一个线程调用一个共享变量 wait() 方法时, 该调用线程会被阻塞挂起。当其他线程调用了该共享对象的 notify 或者 notifyAll 方法时,该线程才返回。
- 调用 wait() 前要事先获取该对象的监视器锁,获取一个共享变量监视器锁的方法:①执行 synchronized 同步代码块时,使用该共享变量作为参数;②调用该共享变量方法时,使用 synchronized 修饰。
- 当前线程调用共享变量的 wait() 方法后只会释放当前共享变量上的锁,持有的其他共享变量的锁不会被释放。
- wait 方法会释放同步监视器,而 sleep 和 yield 方法只是暂停当前线程的执行,不会释放同步监视器。
notify()
- 一个线程调用共享对象的 notify() 方法后,会唤醒一个在该共享变量上调用 wait 系列方法后被挂起的线程。 一个共享变量上可能会有多个线程在等待,具体唤醒哪个等待的线程是随机的。
- 被 notify 方法唤醒的线程必须获取到唤醒它的线程释放的共享变量上的监视器锁后才能返回并继续执行。被唤醒的线程需要和其他线程一起竞争该锁,只有该线程竞争到了共享变量的监视器锁后才可继续执行。
5.suspend和Resume
6.join()和yield()
- thread.Join把指定的线程加入到当前线程,可以将两个交替执行的线程合并为顺序执行的线程。
- 比如在线程B中调用了线程A的Join()方法,直到线程A执行完毕后,才会继续执行线程B。
public class JoinThread extends Thread {
//有参构造器,用于设置该线程的名字
public JoinThread(String name) {
super(name);
}
public void run() {
for (int i = 0; i < 100; i++) {
System.out.println(getName() + " " + i);
}
}
public static void main(String[] args) throws InterruptedException {
// new JoinThread("新线程").start();
for (int i = 0; i < 100; i++) {
if (i == 20) {
JoinThread jt = new JoinThread("join的线程");
jt.start();
jt.join();
}
System.out.println(Thread.currentThread().getName() + " " + i);
}
}
}
- yield方法
7.线程的生命周期(5种状态)
在线程的生命周期中,它会经过新建(New)、就绪(Running)、阻塞(Blocked)和死亡(Dead)五种状态。由于CPU在多条线程之间切换,于是线程状态会多次在就绪和运行之间切换
- 新建状态:当程序使用 new 关键字创建一个线程后,该线程就处于新建状态,此时它仅仅由Java虚拟机分配内存,并初始化其成员变量的值。程序只能对新建状态的线程调用一次 start() 方法。
- 就绪状态:当线程对象调用 start() 方法后,该线程并没有马上执行而是进入就绪状态,Java虚拟机会为其创建方法调用栈和程序计数器,等待获取CPU资源后才会真正处于运行状态。
- 运行状态:处于就绪状态的线程获得CPU后,开始执行 run() 方法的线程执行体,进入运行状态。
- 阻塞状态:线程调用sleep()方法、调用阻塞式IO方法、试图获得同步监视器但该同步监视器正被其他线程持有、在等待某个通知(notify) 时会进入阻塞状态。阻塞状态解除后会重新进入就绪状态,重新等待线程调度器再次调度它。
- 死亡状态:线程 run()或call()方法执行完成、线程抛出一个未捕获的Exception或Error、直接调用线程的stop()方法 时会进入死亡状态。
8.减少线程安全所带来的负面影响
- 不要对线程安全类的所有方法进行同步,只对那些会改变共享资源的方法进行同步。
- 为可变类提供线程不安全和安全两种版本,在单线程环境中使用线程不安全的版本保证性能,在多线程环境中使用线程安全的版本。(比如 StringBuilder 和 StringBuffer)