1.创建和运行线程
每个Java程序在启动时都已经创建了一个线程,叫做主方法线程,也叫主线程。如果想要在主线程之外创建其他的线程,可以有如下几种方式:
方法一,使用Thread
//创建线程对象
//推荐构造方法的参数就是给线程指定名字
Thread t1 = new Thread("t1"){
@Override
public void run() {
System.out.println(111);
}
};
//启动线程
t1.start();
方法二,使用Runnable配合Thread
把线程和任务(要执行的代码分开)
- Thread 代表线程
- Runnable可运行的任务(线程要执行的代码)
Runnable runnable = new Runnable() {
@Override
public void run() {
System.out.println(111);
}
};
Thread thread = new Thread(runnable,"t2");
thread.start();
lambda表达式写法
Runnable runnable =()->System.out.println(111);
Thread thread = new Thread(runnable,"t2");
thread.start();
Thread与Runnable的关系
- 上面两种创建线程的方法中:方法一就是将线程与任务合并在了译器,方法二十八线程和任务分开了。
- 使用Runnable更容易与线程池等高级API相配合
- 使用Runnbale让任务类脱离了Thread的继承体系,变得更加灵活(组合由于继承)
方法三,FutureTask 配合Thread
FutureTask<Integer> task = new FutureTask<>(new Callable<Integer>() {
@Override
public Integer call() throws Exception {
System.out.println("running");
Thread.sleep(1000);
return 1000;
}
});
Thread thread = new Thread(task,"task");
thread.start();
//阻塞等待t1线程执行完成结果的返回
System.out.println(task.get());
不同的操作系统下查看进程线程的方法
windows:
- 任务管理器
- tasklist命令查看进程信息
tasklist | findstr java
- taskkill 杀死进程
taskill /F /PID 29999
Linux
- 查看进程
ps -ef | grep java
- 查看进程的线程信息
top -H -p 4396
- 杀死命令
kill 4396
Java:
- jps查看所有Java进程
- jstack 查看Java进程快照
- jconsole 来查看Java进程中的线程运行情况
线程运行的原理
栈与栈帧
Java虚拟机有堆、栈和方法区组成,其中栈内存就是给线程使用的空间,每启动一次线程,虚拟机就会为其分配一块栈内存
- 每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存
- 每个线程只能有一个活动栈帧,对应着当亲正在执行的那个方法
上下文切换
因为一些原因导致CPU不再执行当前的线程 ,转而执行另外一个线程的代码
原因通常有:
- 线程的 cpu 时间片用完
- 垃圾回收
- 有更高优先级的线程需要运行
- 线程自己调用了sleep、yield、wait、join、park、synchronized、lock等方法。
需要注意的是,当Context Switch 发生时,需要有操作系统保持当前线程的状态,并恢复另一个线程的状态,Java中对应的概念就是程序计数器(Program Counter Register),它的作用就是记住jvm指令的执行地址,是线程私有的。
状态包括程序计数器、虚拟机栈中的每个栈帧信息,比如局部变量、操作数栈、返回地址等
注意:Context Switch 频繁发生会影响性能