java线程

[TOC]

4 运行原理

4.1 栈与栈帧

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

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

比如说,这里就有多个方法调用时的栈帧,每一个栈帧的右边都有其自己对应的变量和属性。如果一个方法执行完了,那么这个栈帧的内存就会被回收,这个是不需要我们自己手动操作的。

PS:我们调试的时候有一个小技巧,就是drop to frame,对应图片中的图标,他的意思是,当你点击这个图标,他会放弃当前栈帧,并且返回上一个调用方法。

image-20210126225902191
image-20210126225840852

4.2 图解运行栈帧

java代码

public class FrameTest {
    public static void main(String[] args) {
        method1(10);
    }

    private static void method1(int x) {
        int y = x + 1;
        Object m = method2();
        System.out.println(m);
    }

    private static Object method2() {
        Object n = new Object();
        return n;
    }
}   
  1. 首先一开始程序会程序栈、方法区和堆,方法区存储的就是方法的内容,main程序运行的时候会有String数组,然后传到main的局部变量表里面
image-20210126232953253
  1. main方法执行第一行method1(10)代码,这个代码指令会放到程序计数器里面(程序计数器记录的就是当前线程需要执行的指令,如果CPU要执行这个线程,其实就是从程序计数器里面拿执行的指令)
image-20210126233402654
  1. 当执行到了method1就会在程序栈里面开辟一个新的程序栈帧

    image-20210126233900325
  2. 当执行到Object m = method2(),这时又会开一个新的程序栈帧

    image-20210126234358464
  3. 当执行完了Object n =new Object(),method2返回时,method2的栈帧会被释放了,同时method2栈帧里面的返回地址回到上一个方法,同时把m指向开辟的Object的堆内存。

    image-20210126234635765
  4. 当执行完method1和main方法也是类似。

4.3 多线程栈和栈帧

package com.bruce.test;

public class FrameTest {
    public static void main(String[] args) {
        Thread t1 = new Thread("t1") {
            @Override
            public void run() {
                method2();
            }
        };
        t1.start();
        method1(30);
    }

    private static void method1(int x) {
        int y = x + 1;
        Object m = method2();
        System.out.println(m);
    }

    private static Object method2() {
        Object n = new Object();
        return n;
    }
}

我们直接看这个的运行情况,就可以看到有两个线程是已经停止了

image-20210128223710073

可见,栈帧是以线程为单位,两者相互独立,里面的变量是相互独立的。

4.4 上下文切换

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

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

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

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

4.4.1 图解上下文切换

image-20210128230204016
  1. 比如说由main线程切换到t1线程的时候,main线程里面的状态信息都会保存起来
  2. CPU会执行t1线程里面的程序计数器里面的指令。

5 线程的常用方法

方法名 static 功能说明 注意
start 启动一个新线程,在新的线程运行run方法中的代码 start方法只是让线程进入就绪,里面代码不一定立刻运行(CPU的时间片还没分给它)。每个线程对象的start方法智能调用一次,如果调用了多次会出现IllegalThreadStateException
run 新线程启动后悔调用的方法 如果在构造Thread对象时传递了Runnable参数,则线程启动后悔调用Runnable中的run方法,否则默认不执行任何操作。但可以创建Thread的子类对象来覆盖默认行为
join 等待线程运行结束
join(long n) 等待线程运行结束,最多等待n毫秒
get() 获取线程长整形的id id唯一
getName() 获取线程名
setName(String) 修改线程名
getPriority() 获取线程优先级
setPriority(int) 修改线程优先级 java中规定线程优先级1~10的整数,较大优先级能提高该线程被CPU调度的概率
getState(0) 获取线程状态 Java中线程状态是用6个enum表示,分别为NEW RUNNABLE,BLOCKED,WAITING,TIMED_WAITING,TERMINATED
isInterrupted() 判断是否被打断 不会清除打断标记
isAlive() 线程存活(还没有运行完毕)
interrupt() 打断线程 如果被打断线程正在sleep,wait,join会导致被打断的线程抛出InterruptedException,并清除打断标记;如果打断的正在运行的线程,则会设置打断标记;park的线程被打断,也会设置打断标记
interrupted() static 判断当前线程是否被打断 会清除打断标记
currentThread(0) static 获取点前正在执行的线程
sleep(long n) static 让当前执行的线程休眠n毫秒,休眠时让出cpu的时间片给其他线程
yield() static 提示线程调度器让出当前线程对CPU的使用 主要是为了测试和调试

二段终止模式

流程图

image-20210125224038357
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,063评论 6 510
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,805评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,403评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,110评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,130评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,877评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,533评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,429评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,947评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,078评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,204评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,894评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,546评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,086评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,195评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,519评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,198评论 2 357

推荐阅读更多精彩内容

  • Java 多线程 学习笔记 一. 线程与进程 进程是一个应用程序,线程是一个进程中的执行场景/执行单元。一个进程可...
    六比特派大锤阅读 276评论 0 0
  • 前言:对于java多线程的资料数不胜数,我们这里只是在学习后简单地总结一下并给出一些小例子,本文适合java初学者...
    LiuHJ阅读 331评论 0 0
  • 进程是指运行中的应用程序,每个进程都有自己独立的地址空间; 线程是进程中执行运算的最小单位,一个进程中可以有多个线...
    SuperFatso阅读 127评论 0 2
  • 进程:正在执行的程序,是一个动态的过程 线程:是进程中用于控制程序执行的控制单元(执行路径,执行情景) 进程中至少...
    宝塔山上的猫阅读 442评论 0 1
  • 首先回顾一下进程(Process)和线程(Thread)的区别: 进程:每个进程都有独立的代码和数据空间(进程上下...
    DN_0915阅读 272评论 0 1