1. 创建和运行线程
方法一,直接使用 Thread
// 构造方法的参数是给线程指定名字,推荐
Thread t1 = new Thread("t1") {
@Override
// run 方法内实现了要执行的任务
public void run() {
log.debug("hello");
}
};
t1.start();
方法二 使用 Runnable 配合 Thread
Runnable r= () -> log.debug("running");
Thread t=new Thread(r,"t2");
t.start();
方法1 是把线程和任务合并在了一起,方法2 是把线程和任务分开了
用 Runnable 更容易与线程池等高级 API 配合
用 Runnable 让任务类脱离了 Thread 继承体系,更灵活
方法三,FutureTask 配合 Thread
FutureTask 能够接收 Callable 类型的参数,用来处理有返回结果的情况
// 创建任务对象
FutureTask<Integer> task3 = new FutureTask<>(() -> {
log.debug("hello");
return 100;
});
// 参数1 是任务对象; 参数2 是线程名字,推荐
new Thread(task3, "t3").start();
// 主线程阻塞,同步等待 task 执行完毕的结果
Integer result = task3.get();
log.debug("结果是:{}", result);
2. 观察多个线程同时运行
主要是理解 线程交替执行,谁先谁后,不由我们控制
@Slf4j(topic = "c.Test3")
public class Test3 {
public static void main(String[] args) {
new Thread(() -> {
while(true) {
log.debug("running...");
}
}, "t1").start();
new Thread(() -> {
while(true) {
log.debug("running...");
}
}, "t2").start();
}
}
t1 t2交替运行
3. 查看进程和线程的方法
Windows
任务管理器
用于查看进程和线程数量,也可以用来结束进程。
命令行工具
tasklist
查看所有进程。
taskkill
杀死指定进程。
Linux
命令行工具
ps -fe
查看所有进程。
ps -fT -p <PID>
查看指定进程(PID)的所有线程。
kill
杀死指定进程。
top
按大写 H 切换是否显示线程。
top -H -p <PID>
查看指定进程(PID)的所有线程。
Java
命令行工具
jps
查看所有 Java 进程。
jstack <PID>
查看指定 Java 进程(PID)的所有线程状态。
图形界面工具
jconsole
用于查看某个 Java 进程中线程的运行情况
4. 原理之线程运行
栈与栈帧
Java Virtual Machine Stacks (Java 虚拟机栈)
我们都知道 JVM 中由堆、栈、方法区所组成,其中栈内存是给谁用的呢?其实就是线程,每个线程启动后,虚拟机就会为其分配一块栈内存。每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存。每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法。
线程上下文切换(Thread Context Switch)
因为以下一些原因导致 cpu 不再执行当前的线程,转而执行另一个线程的代码如:线程的 cpu 时间片用完、垃圾回收、有更高优先级的线程需要运行 或者线程自己调用了 sleep、yield、wait、join、park、synchronized、lock 等方法。
当 Context Switch 发生时,需要由操作系统保存当前线程的状态,并恢复另一个线程的状态,Java 中对应的概念就是程序计数器(Program Counter Register),它的作用是记住下一条 jvm 指令的执行地址,是线程私有的状态包括程序计数器、虚拟机栈中每个栈帧的信息,如局部变量、操作数栈、返回地址等。Context Switch 频繁发生会影响性能。