第1章:并发编程线程基础

目录

  • wait()
  • notify()和notifyAll()
  • join()
  • sleep()
  • yield()
  • 线程中断
    • void interrupt()
    • boolean isInterrupted()
    • boolean interrupted()
    • 示例
  • 守护进程与用户进程
    • 示例
  • 更多

wait()

如果调用wait()方法的线程没有实现获取该对象的监视器锁,则调用wait()方法时线程会抛出IllegalMonitorStateException异常。

一个线程获取一个共享变量的监视器锁的方法

  • 执行synchronized同步代码块时,使用该共享变量作为参数:
synchronized(共享变量) {
    //do something
}
  • 调用该共享变量的方法,并且该方法使用了synchronized修饰:
synchronized void add(int a, int b) {
    //do something
}

注意:一个线程可以挂起状态变成运行状态(也就是被唤醒),即使该线程没有被其他线程调用notify()、notifyAll()方法进行通知,或者被中断,或者等待超时,这就是所谓的虚假唤醒

虚假唤醒在实践中很少发生,但要防患于未然,如下:

synchronized(obj) {
    while(条件不满足) {
        obj.wait();
    }
}

notify()和notifyAll()

notify()会唤醒被阻塞到该共享变量上的一个线程,notifyAll()会唤醒所有在该共享变量上由于调用wait系列方法而被挂起的线程。

join()

...
public static void main(String[] args){
    ...
    thread1.join();
    thread2.join();
    System.out.println("all child thread over!");
}

主线程首先会在调用thread1.join()后被阻塞,等待thread1执行完毕后,thread2开始阻塞,以此类推,最终会等所有子线程都结束后main函数才会返回。

sleep()

sleep()会使线程暂时让出指定时间的执行权,也就是在这期间不参与CPU的调度,但不会释放锁。

yield()

线程调用yield()方法时,实际上是暗示线程调度器当前线程请求让出自己的CPU使用(告诉线程调度器可以进行下一轮的线程调度),但线程调度器可以无条件忽略这个暗示。

线程中断

void interrupt()

设置线程的中断标志为true并立即返回,但线程实际上并没有被中断而会继续向下执行;如果线程因为调用了wait系列函数、join方法或者sleep方法而被阻塞挂起,其他线程调用该线程的interrupt()方法会使该线程抛出InterruptedException异常而返回。

boolean isInterrupted()

检测当前线程是否被中断,是则返回true,否则返回false。

boolean interrupted()

检测当前线程是否被中断,返回值同上,但如果发现当前线程被中断,会清除中断标志;该方法是static方法,内部是获取当前调用线程的中断标志而不是调用interrupted()方法的实例对象的中断标志。

示例

public static void main(String[] args){

    Thread threadOne = new Thread(new Runnable(){
        
          @Override
          public void run() {
              for(;;) {

              }
          }
      });

      //启动线程
      threadOne.start();

      //设置中断标志
      threadOne.interrupt();

      //获取中断标志
      System.out.println("isInterrupted:" + threadOne.isInterrupted() );

      //获取中断标志并重置
      System.out.println("isInterrupted:" + threadOne.interrupted() );

      //获取中断标志并重置
      System.out.println("isInterrupted:" + Thread.interrupted() );

      //获取中断标志
      System.out.println("isInterrupted:" + threadOne.isInterrupted() );

      try{
          threadOne.join();
      }catch(InterruptedException e) {
          System.out.println("interrupted!!!");
      }

      System.out.println("main thread is over");
}

输出为

isInterrupted:true
isInterrupted:false
isInterrupted:false
isInterrupted:true

解析
threadOne.interrupted()一句,由于interrupted()为static方法,相当于执行Thread.interrupted()。

守护进程与用户进程

当最后一个用户进程结束时,JVM会正常退出,而不管当前是否有守护进程。

示例

public static void main(String[] args){

    Thread t = new Thread(new Runnable(){
        
        @Override
        public void run() {
            for(;;) {}
        }
    });

    // t.setDaemon(true);
    t.start();
    System.out.println("main is over");
}

运行后发现虽然输出了"main is over",但ide运行状态红框仍亮着,说明JVM进程未结束,如图:

打开注释后程序则快速退出。

更多

相关笔记:《Java并发编程之美》阅读笔记

你可能感兴趣的:(第1章:并发编程线程基础)