Java 线程

发布时间 2023-05-28 22:30:02作者: archaique

栈与栈帧 Java Virtual Machine Stacks (Java 虚拟机栈)

  • 每个线程启动后,虚拟机就会为其分配一块栈内存,是线程私有的。
  • 每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存;每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法

 

 

线程上下文切换(Thread Context Switch)

因为以下一些原因导致 cpu 不再执行当前的线程,转而执行另一个线程的代码

  • 线程的 cpu 时间片用完
  • 垃圾回收
  • 有更高优先级的线程需要运行
  • 线程自己调用了 sleep、yield、wait、join、park、synchronized、lock 等方法

当 Context Switch 发生时,需要由操作系统 保存当前线程的状态,并恢复另一个线程的状态,Java 中对应的概念 就是程序计数器(Program Counter Register),它的作用是记住下一条 jvm 指令的执行地址,是线程私有的

  • 状态包括程序计数器、虚拟机栈中每个栈帧的信息,如局部变量、操作数栈、返回地址等。
  • Context Switch 频繁发生会影响性能

 

start 与 run

调用 Run

public static void main(String[] args) {
   Thread t1 = new Thread("t1") {
     @Override
     public void run() {
       log.debug(Thread.currentThread().getName());
       // 读 MP4 文件,要花很多时间       FileReader.read(Constants.MP4_FULL_PATH);      }   };   t1.run();   log.debug(
"do other things ..."); }

输出

19:39:14 [main] c.TestStart - main
19:39:14 [main] c.FileReader - read [1.mp4] start ...
19:39:18 [main] c.FileReader - read [1.mp4] end ... cost: 4227 ms
19:39:18 [main] c.TestStart - do other things ...

程序仍在 main 线程运行, FileReader.read() 方法调用还是同步的,没有新起一个线程异步执行

调用 Start

将上述代码的 t1.run() 改为

t1.start();

输出

19:41:30 [main] c.TestStart - do other things ...
19:41:30 [t1] c.TestStart - t1
19:41:30 [t1] c.FileReader - read [1.mp4] start ...
19:41:35 [t1] c.FileReader - read [1.mp4] end ... cost: 4542 ms

程序在 t1 线程运行, FileReader.read() 方法调用是异步的。

但 start 方法只是让线程进入就绪,里面代码不一定立刻运行(CPU 的时间片可能还没分给它)

每个线程对象的 start方法只能调用一次,如果调用了多次会出现 IllegalThreadStateException。

System.out.printlin(t1.getState());
t1.start();
System.out.printlin(t1.getState());

NEW 状态 ——>   Runable 状态