当前位置: 首页 > article >正文

运行打印Hello World启动了多少线程?

序言

看网上说阿里二面问到了一个看似最简单且没有标准答案的一个问题,所有学习编程都是从打印hello World开始的,那运行打印启动了多少个线程?

启动了多少线程?

在运行一个简单的 “Hello World” 程序时,启动的线程数量取决于编程语言、运行时环境和操作系统。

就编程语言而言,C语言、C++、Java、Python、Go、JavaScript都是不同的,且最少有一个线程。如果不考虑编程语言的话,就Java语言,有多少个线程也不是确定的。需要看JDK的版本,不同JDK版本启动的线程数量也可能是不同的。

比如下面这个简单的Java程序,怎么看它运行了多少个线程呢?

public class Main {public static void main(String[] args) {System.out.println("Hello World");}
}

假设当前电脑JDK版本是Java 8,
在这里插入图片描述

strace命令

方法有很多,如果你是linux环境,使用strace·(system trace是 Linux 系统中一个功能强大的调试、分析、诊断工具,可以对系统调用和信号传递的跟踪结果进行分析)命令进行分析,

  • -c:统计每个系统调用的时间、次数和错误,并在程序退出时报告摘要。
  • -D:将跟踪进程作为分离的孙进程运行,减少strace的可见效果。
  • -d:输出strace关于标准错误的调试信息。
  • -f:跟踪由fork调用所产生的子进程。
  • -t:在输出中的每一行前加上时间信息。
  • -v:输出所有的系统调用,包括环境变量和状态信息。
  • -o:将跟踪结果输出到文件中‌。

在这里插入图片描述
下面的h1就是分析的结果文件,后缀就是对应的线程编号,即下面有很多个线程,less h1.11767可以查看对应的线程内容
在这里插入图片描述
上面代表的是linux操作系统的系统调用日志。

JProfiler工具

JProfiler(Java Profiler)是IDEA自带的Java性能分析工具。左边有几个线程ID,比如线程ID=13571的线程,JavaThread::run() -> CompileBroker::invoke_compiler_on_method(CompileTask*),它是JVM中JIT即时编译器线程,用于在程序运行时将代码转换为机器码。
在这里插入图片描述

下面线程id = 17155的线程,JavaThread::run() -> Compile::Compile(ciEnv*, C2Compiler*, ciMethod*, int, bool, bool, bool),它是进行编译的线程,入参有编译器接口环境指针,C2编译器指针,要编译的方法等参数,
在这里插入图片描述

destroyjavavm线程,它是用于销毁JVM实例的线程,
在这里插入图片描述
Monitor Ctrl-Break监控线程,用于监控控制台中断信号,
在这里插入图片描述
还有Finalizer 线程是一个由 JVM 自动创建的守护线程(Daemon Thread),负责执行对象的 finalize() 方法。它是 Java 垃圾回收(Garbage Collection, GC)机制的一部分。

Thread.getAllStackTraces方法

如下图,该方法获取了所有活动线程的堆栈跟踪信息。

  • Thread[Signal Dispatcher,9,system]=[Ljava.lang.StackTraceElement;@579bb367
  • Thread[main,5,main]=[Ljava.lang.StackTraceElement;@1de0aca6,
  • Thread[Monitor Ctrl-Break,5,main]=[Ljava.lang.StackTraceElement;@255316f2,
  • Thread[Finalizer,8,system]=[Ljava.lang.StackTraceElement;@41906a77,
  • Thread[Reference Handler,10,system]=[Ljava.lang.StackTraceElement;@4b9af9a9
    在这里插入图片描述
    其中Reference Handler线程是Java虚拟机(JVM)中的一个特殊线程,用于处理垃圾回收过程中的对象引用。在Java中,对象引用可能会被垃圾回收器回收,但是如果仍然有其他对象引用这些对象,那么这些对象就不会被回收。Reference Handler线程的作用就是处理这些被引用但不再被使用的对象,以避免内存泄漏。

Reference Handler线程的主要功能是处理四种类型的引用:

  • SoftReference:软引用,只有在内存不足时才会被回收。
  • WeakReference:弱引用,在下一次垃圾回收时会被回收。
  • PhantomReference:虚引用,用于在对象被回收之前执行一些操作。
  • FinalReference:最终引用,用于处理具有finalize方法的对象。

进程、线程和协程的区别

  • 进程‌:系统资源分配的最小单位,每个进程拥有独立的内存空间、文件描述符等资源,进程间完全隔离。
  • 线程‌:运行在进程内部,共享进程的内存空间和文件句柄等资源,但每个线程有自己的栈和局部变量,线程间通过共享资源直接通信。
  • 协程‌:在单线程内通过用户态调度实现并发,不涉及系统资源分配,仅在用户态切换执行流,因此没有独立的内存。

进程(Process)

进程是程序在计算机上运行的一个实例,那一个应用是否是一个进程呢?大多数情况如此,但有些应用中可能包含多个进程,比如VS Code就包含主进程、渲染进程和插件进程。

线程(Thread)

线程是操作系统内核调度的基本单位,线程的创建、销毁、多线程之间的切换都需要操作系统的内核完成。
在这里插入图片描述

协程(Coroutine)

而协程则是利用同一个调度线程去实现协程之间的切换,从而避免了CPU在内核态和用户态之间的来回切换,所以协程的效率比线程要高,比如Go语言天然支持协程,它在协程的帮助下可以轻松实现百万并发。
在这里插入图片描述
那为什么Java一开始就没有用协程的设计呢?原因是由于Java诞生于1995年,多线程模型已经成熟,且被各种操作系统广泛支持。而协程还没有标准化的支持,在不同操作系统上实现方式差距很大,兼容性也不好,所以Java就选择了更加成熟的线程模型。

之后Java在多线程的路上越走越深,为了解决频繁创建和销毁线程导致的性能开销,衍生出了线程池技术。
在这里插入图片描述
但是始终无法解决下面两个问题:

  • 1、线程太多,会导致CPU频繁上下文切换,性能开销大;
  • 2、线程太少,系统吞吐量又不足;

所以衍生出了多线程参数调优:

  • 1、识别适合并行执行的任务;
  • 2、保证线程安全;
  • 3、避免死锁和竞态条件;
  • 4、合理分配核心线程数;
  • 5、优化线程间通信;
  • 6、持续监控和优化;

虚拟线程(Virtual Threads)

JVM)在处理多线程应用时,会使用线程上下文来执行各个线程的任务。然而,在某些情况下,线程上下文可能会成为性能瓶颈。为了解决这个问题,JDK19引入了一项新特性——虚拟线程(Virtual Threads,简称VTs)。虚拟线程是一种在JVM内部实现的并行执行机制,可以提高多线程应用的性能。

与传统的线程相比,虚拟线程具有以下特点:

  1. 无锁竞争:虚拟线程在执行过程中不需要进行锁竞争,这意味着它可以显著降低线程切换的开销,从而提高程序的运行效率。

  2. 无需线程上下文:虚拟线程在执行过程中不需要线程上下文,这意味着它可以更好地利用处理器资源,提高程序的并行执行能力。

  3. 更低的内存开销:虚拟线程的实现方式相对于线程更为轻量级,因此它在内存占用方面具有更小的开销。

看网上的文章是使用@RunOnVirtualThread注解通过AOP去使用虚拟线程,目前开发使用的场景还没有使用过虚拟线程,也不清楚实际的效率提升是多少,需要各位工程师在实践上去测试。

相关文章:

运行打印Hello World启动了多少线程?

序言 看网上说阿里二面问到了一个看似最简单且没有标准答案的一个问题,所有学习编程都是从打印hello World开始的,那运行打印启动了多少个线程? 启动了多少线程? 在运行一个简单的 “Hello World” 程序时,启动的线…...

C++项目中调用C#DLL的的方式

C项目中调用C#DLL的的方式 方法一:使用COM技术方法二:使用C/CLI方法三:使用P/Invoke(适用于C#导出非托管接口) 在C中调用C#编写的DLL,通常需要借助COM(Component Object Model&#…...

咳嗽止咳药笔记250526 , 磷酸苯丙哌林 , 喷托维林 , 右美沙芬

咳嗽止咳药笔记250526 止咳药的种类较多,根据作用机制可分为中枢性止咳药、外周性止咳药、祛痰药、抗组胺药及中成药等。以下是具体分类及效果分析: 一、中枢性止咳药 可待因 效果:直接抑制延髓咳嗽中枢,镇咳作用强且迅速&#x…...

vue pinia 独立维护,仓库统一导出

它允许您跨组件/页面共享状态 持久化 安装依赖pnpm i pinia-plugin-persistedstate 将插件添加到 pinia 实例上 pinia独立维护 统一导出 import { createPinia } from pinia import piniaPluginPersistedstate from pinia-plugin-persistedstateconst pinia creat…...

网络的协议和标准

网络的协议和标准 OSI参考模型 应用层 报文 网关 表示层 报文 会话层 报文 传输层 报文 网络层 数据包 路由器 数据链路层 帧 网桥交换机 物理层 位 中继器 集线器 TCP/IP协议簇 逻辑地址:每台设备都有一个ip地址 一个ip地址包包含网络号 子网络号 主机号可…...

十六进制字符转十进制算法

十六进制与十进制对照 十六进制十进制00112233445566778899A10B11C12D13E14F15 十六进制与十进制区别 十六进制是满16进1,十进制是满10进1,这里要注意下区别,16进制的字符里面为什么是0-9没有10,这里面进了一位,表示…...

跟Gemini学做PPT:汇报背景图寻找指南

PPT 汇报背景图寻找指南 既然前端功能已经完善,现在可以专注于汇报了。对于 PPT 背景图,你有几个选择: 1. 内置模板和主题: 优点: 最简单、快速,PowerPoint、Keynote、Google Slides 等演示软件都内置了…...

java交易所,多语言,外汇,黄金,区块链,dapp类型的,支持授权,划转,挖矿(源码下载)

目前这套主要是运营交易所类型的,授权的会贵点,编译后的是可以直接跑的,图片也修复了,后门也扫了 都是在跑的项目支持测,全开源 源码下载:https://download.csdn.net/download/m0_66047725/90887047 更多…...

(已开源-CVPR2024) RadarDistill---NuScenes数据集Radar检测第一名

本文介绍一篇Radar 3D目标检测模型:RadarDistill。雷达数据固有的噪声和稀疏性给3D目标检测带来了巨大挑战。在本文中,作者提出了一种新的知识蒸馏(KD)方法RadarDistill,它可以通过利用激光雷达数据来提高雷达数据的表征。RadarDistill利用三…...

【MySQL】 数据库基础数据类型

一、数据库简介 1.什么是数据库 数据库(Database)是一种用于存储、管理和检索数据的系统化集合。它允许用户以结构化的方式存储大量数据,并通过高效的方式访问和操作这些数据。数据库通常由数据库管理系统(DBMS)管理&…...

中小企业AI算力如何选?【显卡租赁】VS【自建服务器】

对于中小企业而言和科研单位来讲,AI算力的选择需综合考虑成本、灵活性、数据安全和技术迭代风险等因素。以下是显卡租赁与自建服务器的对比分析,帮助中小企业做出最优决策: 1. 成本对比 自建服务器 高昂的前期投入:搭建一个中等规…...

OpenHarmony 4.1版本应用升级到5.0版本问题记录及解决方案

目录 1. ERROR: ArkTS:ERROR File: E:/Hap/applications_contacts-OpenHarmony-5.0.0-Release/entry/src/main/ets/Application/MyAbilityStage.ts:33:9 No overload matches this call. Overload 1 of 4, (slot: NotificationSlot): Promise, gave the following error. …...

std::initialzer_list 与花括号{}数据列表

author: hjjdebug date: 2025年 05月 22日 星期四 15:50:23 CST descrip: std::initialzer_list 与花括号{}数据列表 文章目录 1.{数值列表}是什么?1.1 数组初始化 时 , 称为数组初始化列表1.2. 当用于容器时, 称为容器初始化列表1.3. 对于结构体或类,{…...

萤石云实际视频实时接入(生产环境)

萤石云视频接入 本示例可用于实际接入萤石云开放平台视频,同时支持音频输入和输出。 实际优化内容 1.动态获取token 2.切换各公司和车间时,自动重新初始化播放器 let EZUIKit null; // 第三方库引用 let EZUIKitPlayers []; // 播放器实例数组 le…...

QT中常用的类

Qt 是一个功能强大的跨平台框架,提供了丰富的类库来开发 GUI 和应用程序。以下是 ​​Qt 中常用的核心类​​,按模块分类整理: ​​1. GUI 和窗口管理​​ 类名用途示例场景​​QWidget​​所有 GUI 控件的基类(按钮、窗口等&…...

Docker系列(四):容器操作全栈技术指南 --- 从入门到生产级管控

引言 本指南以全链路视角拆解Docker技术栈,通过四大核心模块构建从入门到进阶的知识体系,助您系统性掌握容器化落地的关键能力。 容器生命周期管理(一)从创建、启停到资源清理,夯实容器操作的基础语法与核心场景&…...

poppler_path 是用于 Python 库如 pdf2image 进行 PDF 转换时

poppler_path 是用于 Python 库如 pdf2image 进行 PDF 转换时指定 Poppler 可执行文件路径的参数。为了让程序正常工作,需要先安装 Poppler,并配置环境变量或在代码中设置 poppler_path。 以下是 Poppler 的安装与环境变量配置方法,按操作系…...

鸿蒙OSUniApp 开发的多图浏览器组件#三方框架 #Uniapp

使用 UniApp 开发的多图浏览器组件 在移动应用开发中,图片浏览器是非常常见且实用的功能,尤其是在社交、资讯、电商等场景下,用户对多图浏览体验的要求越来越高。随着 HarmonyOS(鸿蒙)生态的不断壮大,开发…...

MongoDB 错误处理与调试完全指南:从入门到精通

在当今数据驱动的世界中,MongoDB 作为最流行的 NoSQL 数据库之一,因其灵活的数据模型和强大的扩展能力而广受开发者喜爱。然而,与任何复杂系统一样,在使用 MongoDB 过程中难免会遇到各种错误和性能问题。本文将全面介绍 MongoDB 的…...

React从基础入门到高级实战:React 核心技术 - 表单处理与验证深度指南

React 表单处理与验证深度指南 在现代 Web 应用中,表单是用户与应用交互的核心方式之一。无论是注册、登录、结账还是数据提交,表单都扮演着至关重要的角色。React 作为一款流行的前端框架,提供了多种处理表单的工具和方法,帮助开…...

【C++】stack,queue和priority_queue(优先级队列)

文章目录 前言一、栈(stack)和队列(queue)的相关接口1.栈的相关接口2.队列的相关接口 二、栈(stack)和队列(queue)的模拟实现1.stack的模拟实现2.queue的模拟实现 三、priority_queu…...

ubuntu中上传项目至GitHub仓库教程

一、到github官网注册用户 1.注册用户 地址:https://github.com/ 2.安装Git 打开终端,输入指令git,检查是否已安装Git 如果没有安装就输入指令 sudo apt-get install git 二、上传项目到github 1.创建项目仓库 进入github主页,点击号…...

[Java实战]Spring Boot整合达梦数据库连接池配置(三十四)

[Java实战]Spring Boot整合达梦数据库连接池配置(三十四) 一、HikariCP连接池配置(默认) 1. 基础配置(application.yml) spring:datasource:driver-class-name: dm.jdbc.driver.DmDriverurl: jdbc:dm://…...

windows 下用yolov5 训练模型 给到opencv 使用

windows 使用yolov5训练模型,之后opencv加载模型进行推理。 一,搭建环境 安装 Anaconda 二,创建虚拟环境并安装yolov5 conda create -n yolov5 python3.9 -y conda activate yolov5 git clone https://github.com/ultralytics/yolov5 cd …...

Spark集群架构解析:核心组件与Standalone、YARN模式深度对比(AM,Container,Driver,Executor)

一、核心组件定义与关系拆解 1. ApplicationMaster(AM) 定义:YARN 框架中的应用管理器,每个应用程序(如 Spark 作业)对应一个 AM。职责: 向 YARN 的 ResourceManager 申请资源(Con…...

Linux Kernel调试:强大的printk(二)

前言 如果你对printk的基本用法还不熟悉,请先阅读: Linux Kernel调试:强大的printk(一) 上一篇Linux Kernel调试:强大的printk(一)我们介绍了printk的基础知识和基本用法&#xf…...

Kafka Kraft模式集群 + ssl

文章目录 启用集群资源规划准备证书创建相关文件夹配置文件启动各Kafka节点 故障转移测试spring boot集成 启用集群 配置集群时关键就是提前梳理好需要的网络资源,完成对应server.properties文件的配置。在执行前先把这些梳理好,可以方便后面的配置&…...

[crxjs]自己创建一个浏览器插件

参考官方 https://crxjs.dev/vite-plugin/getting-started/vue/create-project 按照流程操作会失败的原因 是因为跨域的问题, 在此处添加 server: {host: "localhost",port: 5173,cors: true,headers: {"Access-Control-Allow-Origin": "*",}…...

类的设计模式——单例、工厂以及建造者模式

1.单例模式 1.1 饿汉模式 单例模式:一个类只能创建一个对象,这个设计模式可以保证系统中该类只有一个实例,并提供一个访问它的全局访问点,该实例被所有程序模块共享。 饿汉模式指在程序初始化时就创建一个唯一的实例对象。适用…...

STM32之看门狗(IWDG)

一、看门狗外设的原理与应用 背景说明 随着单片机的发展,单片机在家用电器、工业自动化、生产过程控制、智能仪器仪表等领域的应用越来越广泛。然而处于同一电力系统中的各种电气设备通过电或磁的联系彼此紧密相连,相互影响,由于运行方式的…...