【多线程3:基础原理】

简介: 【多线程3:基础原理】

【多线程3:基础原理】

01.栈与栈帧

我们都知道 JVM 中由堆、栈、方法区所组成,其中栈内存是给谁用的呢?其实就是线程,每个线程启动后,虚拟机就会为其分配一块栈内存。

每个栈由多个栈帧(Frame)组成,对应着每次方法调用时所占用的内存 每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法

简而言之一个线程有一个栈,一个方法有一个栈帧

02.单个线程运行原理

一个简单的例子

,public class Test {
    public static void main(String[] args) {
        fun1();
    }
    static void fun1(){
        int i=fun2();
        System.out.println("我是fun1");
        System.out.println(i);
    }
    static int fun2(){
        System.out.println("我是fun2");
        int j = 1;
        return j;
    }
}

结果

我是fun2
我是fun1
1

文字解释

这个程序只有一个主线程也就是一个栈,有三个方法main、fun1、fun2也就是三个栈帧,栈帧入栈方式存入栈中,以出栈方式弹出栈中,当一个方法运行结束就从栈中弹出并释放内存,因为栈的特性是后进先出,所以对于这个程序,入栈顺序为mian、fun1、fun2,出栈顺序为fun2、fun1、main,这就是结果的由来。

debug解释

202206201644772.gif

可以看出程序的运行过程

03.多线程运行原理

例子

public class Test2 {
    public static void main(String[] args) {
        Thread t = new Thread(()->{
            fun1("t1");
        });
        t.setName("t1");
        t.start(); // 副线程

        fun1("main"); // 主线程
    }

    static void fun1(String s){
        int i=fun2(s);
        System.out.println("我是"+s+" fun1");
        System.out.println(i);
    }
    static int fun2(String s){
        System.out.println("我是"+s+" fun2");
        int j = 1;
        return j;
    }
}

结果

我是main fun2
我是t1 fun2
我是t1 fun1
1
我是main fun1
1

因为多线程是的运行是由cpu控制的,所以结果是随机的,上面的结果只是一种情况

文字解释

程序有两个线程,故有两个栈,两个栈相互独立,每个栈都有自己的栈帧,即 main线程的栈帧为main、fun1、fun2,t1线程的栈帧为 t1、fun1、fun2。

需要注意的点是,cpu在运行多线程会随机切换线程,但会保存线程的状态。

debug解释

202206201734870.gif

可以看出两个线程是在独立运行的,且切换到另一个线程时会保存原线程状态。

04.线程上下文切换

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

线程的 cpu 时间片用完

垃圾回收

有更高优先级的线程需要运行

线程自己调用了 sleep、yield、wait、join、park、synchronized、lock 等方法

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

状态包括程序计数器、虚拟机栈中每个栈帧的信息,如局部变量、操作数栈、返回地址等

上下文切换 频繁发生会影响性能

目录
相关文章
|
6天前
|
Java Linux 调度
硬核揭秘:线程与进程的底层原理,面试高分必备!
嘿,大家好!我是小米,29岁的技术爱好者。今天来聊聊线程和进程的区别。进程是操作系统中运行的程序实例,有独立内存空间;线程是进程内的最小执行单元,共享内存。创建进程开销大但更安全,线程轻量高效但易引发数据竞争。面试时可强调:进程是资源分配单位,线程是CPU调度单位。根据不同场景选择合适的并发模型,如高并发用线程池。希望这篇文章能帮你更好地理解并回答面试中的相关问题,祝你早日拿下心仪的offer!
25 6
|
5月前
|
安全 Java 数据库
一天十道Java面试题----第四天(线程池复用的原理------>spring事务的实现方式原理以及隔离级别)
这篇文章是关于Java面试题的笔记,涵盖了线程池复用原理、Spring框架基础、AOP和IOC概念、Bean生命周期和作用域、单例Bean的线程安全性、Spring中使用的设计模式、以及Spring事务的实现方式和隔离级别等知识点。
|
5月前
|
编解码 网络协议 API
Netty运行原理问题之Netty的主次Reactor多线程模型工作的问题如何解决
Netty运行原理问题之Netty的主次Reactor多线程模型工作的问题如何解决
|
4月前
|
存储 缓存 Java
什么是线程池?从底层源码入手,深度解析线程池的工作原理
本文从底层源码入手,深度解析ThreadPoolExecutor底层源码,包括其核心字段、内部类和重要方法,另外对Executors工具类下的四种自带线程池源码进行解释。 阅读本文后,可以对线程池的工作原理、七大参数、生命周期、拒绝策略等内容拥有更深入的认识。
178 29
|
3月前
|
Java 编译器 程序员
【多线程】synchronized原理
【多线程】synchronized原理
72 0
|
3月前
|
Java 应用服务中间件 API
nginx线程池原理
nginx线程池原理
48 0
|
5月前
|
存储 NoSQL Java
线程池的原理与C语言实现
【8月更文挑战第22天】线程池是一种多线程处理框架,通过复用预创建的线程来高效地处理大量短暂或临时任务,提升程序性能。它主要包括三部分:线程管理器、工作队列和线程。线程管理器负责创建与管理线程;工作队列存储待处理任务;线程则执行任务。当提交新任务时,线程管理器将其加入队列,并由空闲线程处理。使用线程池能减少线程创建与销毁的开销,提高响应速度,并能有效控制并发线程数量,避免资源竞争。这里还提供了一个简单的 C 语言实现示例。
107 6
|
4月前
|
存储 缓存 Java
JAVA并发编程系列(11)线程池底层原理架构剖析
本文详细解析了Java线程池的核心参数及其意义,包括核心线程数量(corePoolSize)、最大线程数量(maximumPoolSize)、线程空闲时间(keepAliveTime)、任务存储队列(workQueue)、线程工厂(threadFactory)及拒绝策略(handler)。此外,还介绍了四种常见的线程池:可缓存线程池(newCachedThreadPool)、定时调度线程池(newScheduledThreadPool)、单线程池(newSingleThreadExecutor)及固定长度线程池(newFixedThreadPool)。
|
5月前
|
存储 Java
线程池的底层工作原理是什么?
【8月更文挑战第8天】线程池的底层工作原理是什么?
132 8
|
4月前
|
安全 Java API
Java线程池原理与锁机制分析
综上所述,Java线程池和锁机制是并发编程中极其重要的两个部分。线程池主要用于管理线程的生命周期和执行并发任务,而锁机制则用于保障线程安全和防止数据的并发错误。它们深入地结合在一起,成为Java高效并发编程实践中的关键要素。
42 0