并发与多线程04——线程的基本操作和线程的生命周期

本文深入探讨了线程控制的各种方法,包括线程的创建、终止、中断、等待和通知机制,以及如何通过suspend和resume、join和yield等方法控制线程执行。此外,还详细解析了线程的生命周期,从新建、就绪、运行、阻塞到死亡的全过程,以及如何优化线程安全,避免负面影响。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值