• JUC并发编程与源码分析笔记02-线程基础知识复习


    先拜拜大神

    JUC包的作者是:Doug Lea

    为什么学习并用好多线程极其重要

    硬件方面

    摩尔定律失效。

    软件方面

    面试更有底气,充分利用多核处理器的优势,提高程序性能,提高程序吞吐量,完成异步+回调等生产需求。

    弊端及问题

    线程安全问题、线程锁问题、线程性能问题。

    从start一个线程说起

    public class Test {
        public static void main(String[] args) {
            new Thread(() -> {
                System.out.println("Test.main");
            }, "myThread").start();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    Java线程理解以及openjdk中的实现

    Java最底层的native方法,是由C++实现的,可以到openjdk查看,下载地址在这里,我们需要关注的目录是:openjdk\src\hotspot\share\runtime。

    更加底层的C++源码阅读

    Java线程通过start方法启动执行,主要内容在native的start0()
    里,OpenJDK中Thread.java对应的就是Thread.c,start0就是JVM_StartThread。

    Java多线程相关概念

    1把锁

    synchronized。

    2个并

    并发(concurrent)

    某一时刻,在同一实体上的多个事件,在一台处理器上同时处理多个任务。

    并行(parallel)

    某一时刻,在不同实体上的多个事件,在不同处理器上同时处理多个任务。

    并发vs并行

    3个程

    进程

    简单的说,在系统中运行的一个应用程序就是一个进程,每一个进程都有它自己的内存空间和系统资源。

    线程

    也被称为轻量级进程,在同一个进程内会有1个或多个线程,是大多数操作系统进行时序调度的基本单元。

    管程

    俗称Monitor(监视器),是我们平时所说的锁,比如synchronized锁。
    Monitor其实是一种同步机制,它的义务是保证同一时间只有一个线程可以访问被保护的数据和代码,JVM中同步是基于进入和退出监视器对象来实现的,每个对象实例都有一个Monitor对象。Monitor对象和Java对象一同创建并销毁,底层由C++实现。

    用户线程和守护线程

    Java线程分为用户线程和守护线程

    默认情况下,指的都是用户线程。

    用户线程(User Thread)

    是系统的工作线程,它会完成这个程序需要完成的业务操作。

    守护线程(Daemon Thread)

    是一种特殊的线程,是为其它线程服务的,在后台默默地完成一些系统性的服务,比如:垃圾回收线程。
    守护线程作为一个服务线程,没有服务对象就没有必要继续运行了,如果用户线程全部结束了,意味着程序需要完成的业务操作已经结束了,系统可以退出了,所以假如当系统只剩下守护线程的时候,Java虚拟机会自动退出。

    线程的daemon属性

    通过java.lang.Thread#isDaemon方法判断是否是守护线程。

    code展示

    public class Test {
        public static void main(String[] args) {
            Thread myThread = new Thread(() -> {
                System.out.println(Thread.currentThread().getName() + ":" + (Thread.currentThread().isDaemon() ? "守护线程" : "用户线程"));
                while(true);
            }, "myThread");
            myThread.setDaemon(true);
            myThread.start();
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println(Thread.currentThread().getName() + "线程结束");
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16

    留意myThread.setDaemon(true);语句,当myThread是用户线程的时候,main线程结束,但是myThread还在运行,当myThread是守护线程的时候,main线程结束,此时已经没有其他用户线程了,所以myThread也会结束。

    小总结

    如果用户线程全部结束意味着程序需要完成的业务操作已经结束了,守护线程随着JVM一同结束工作,setDaemon(true)方法必须在start()之前设置,否则报IllegalThreadStateException异常。

  • 相关阅读:
    齐博X1-栏目的终极方法get_sort
    正则表达式二
    软件工程师真的只是编码吗?不,他有10个隐蔽的工作
    MFC 学习笔记
    虚拟机VMware Workstation Pro安装配置使用服务器系统ubuntu-22.04.3-live-server-amd64.iso
    idea 无法识别maven的解决
    android获取进程内存使用信息、一键加速(内存清理)与进程重要级别解析
    宇视网络视频录像机人机实况画面偏色排查步骤
    Go语言高阶:Reflection反射与Files操作 详细示例教程
    qlib的工作流管理:mlflow机器学习生命周期管理平台
  • 原文地址:https://blog.csdn.net/qq_36059561/article/details/128052653