Java并发编程之Java线程 上

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 频繁发生会影响性能。

  • 5
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值