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

线程和进程

文章目录

  • 进程和线程
    • 进程
    • 线程
    • 案例
  • 时间片概念
  • 调度方式
  • 线程的创建和启动
    • 第一种创建方式
    • 第二种创建方式(匿名内部类)
    • 第三种创建方式(Runnable接口)
    • main线程和t线程之间的关系
  • 线程的名字
  • 线程的优先级
  • 线程状态


进程和线程

进程

在计算机中,进程代表了内存中正在运行的应用程序,计算机中的资源(cpu 内存 磁盘 网络等),会按照需求分配给每个进程,从而这个进程对应的应用程序就可以使用这些资源了。

在操作系统中,启动一个应用程序的时候,会有一个或多个进程同时被创建,这些进程其实就表示了当前这个应用程序,在系统中的资源使用情况以及程序运行的情况。如果关闭这个进程,那么对应的应用程序也就关闭了。
所以,进程就是在系统中,运行一个应用程序的基本单位。

线程

线程是进程中的一个代码执行单元,负责当前进程中代码程序的执行,一个进程中有一个或多个线程。
当一个进程中启动了多个线程去分别执行代码的时候,这个程序就是多线程程序。

在这里插入图片描述

案例

例如,当前我们去运行一个类的时候,会先启动JVM,这个JVM对于计算机来讲,就是一个
应用程序,所以同时系统中也会启动一个进程和这个JVM对应:

public  class hello{public static void main(String[] args)throws Exception{System.out.println("hello");long time = 1000*100L;Thread.sleep(time);System.out.println("world");}
}

我们运行这个代码:
在这里插入图片描述

注意,代码中,输出hello之后,JVM并没有直接结束,而是让当前线程去休眠了100秒,所以这个时候JVM还在运行着,我们可以在任务管理器中,看到JVM对应的进程了

在这里插入图片描述

除此之外,我们还可以使用JDK中提供的工具,来查看JVM当前的运行情况,在cmd中输入jconsole,然后连接:
在这里插入图片描述
在这里插入图片描述

这里通过JDK自带的jconsole工具,可以检测到当前运行Hello这个类的时候,JVM的运行情况,包含内存的使用、线程的运行状态、类的加载等信息.

例如,查看当前JVM中执行main方法线程信:

在这里插入图片描述
(注意,这线程的名字就叫main,它是任务就是调用执行我们类中的main方法,所以,是一个名字叫main的线程,调用执行我们代码中的main方法)

时间片概念

时间片,当前一个线程要使用CPU的时候,CPU会分配给这个线程一小段时间(毫秒级别),这段时间就叫做时间片,也就是该线程允许使用CPU运行的时间,在这个期间,线程拥有CPU的使用权。

如果在一个时间片结束时,线程还在运行,那么这时候,该线程就需要停止运行,并交出CPU的使用
权,然后等待下一个CPU时间片的分配。

(在宏观上,一段时间内,我们感觉两个线程在同时运行代码,其实在微观中,这俩个线程在使用
一个CPU的时候,它们是交替着运行的,每个线程每次都是运行一个很小的时间片,然后就交出CPU使用权,只是它们俩个交替运行的速度太快了,给我们的感觉,好像是它们俩个线程在同时运行。)

调度方式

当俩个或多个线程使用一个CPU来运行代码的时候,在操作系统的内核中,就会有相应的算法来控制线程获取CPU时间片的方式,从而使得这些线程可以按照某种顺序来使用CPU运行代码,这种情况被称为线程调用。

常见的调度方式有两种:

  • 时间片轮转
    所有线程轮流使用 CPU 的使用权,平均分配每个线程占用 CPU 的时间。
  • 抢占式调度
    系统会让优先级高的线程优先使用 CPU(提高抢占到的概率),但是如果线程的优先级相同,那么会随机选择一个线程获取当前CPU的时间片。

(JVM中的线程,使用的为抢占式调度。)

线程的创建和启动

java.lang.Thread 是java中的线程类,所有的线程对象都必须是Thread类或其子类的实例。

每个线程的作用,就是完成我们给它指定的任务,实际上就是执行一段我们指定的代码。我们只需要在Thread 类的子类中重写 run 方法,把执行的代码写入到run方法中即可,这就是线程的执行任务

Java中通过继承Thread类来创建并启动一个新的线程的步骤如下:

  1. 定义 Thread 类的子类(可以是匿名内部类),并重写 Thread 类中的 run 方法, run 方法中的
    代码就是线程的执行任务
  2. 创建 Thread 子类的对象,这个对象就代表了一个要独立运行的新线程
  3. 调用线程对象的 start 方法来启动该线程

第一种创建方式

package shixun;public class myTest {public static void main(String[] args) {MyThread myThread=new MyThread();myThread.start();}
}class MyThread extends Thread{@Overridepublic void run() {int x=0;while(true){String name = Thread.currentThread().getName();System.out.println("hello, ["+name+"] - "+(x++));try{Thread.sleep(1000);}catch(Exception e){e.printStackTrace();}}}
}

在这里插入图片描述

第二种创建方式(匿名内部类)

package shixun;public class myTest {public static void main(String[] args) {Thread thread=new Thread(){@Overridepublic void run() {int x=0;while(true){String name = Thread.currentThread().getName();System.out.println("hello, ["+name+"] - "+(x++));try{Thread.sleep(1000);}catch(Exception e){e.printStackTrace();}}}};thread.start();}
}

在这里插入图片描述

第三种创建方式(Runnable接口)

package shixun;public class myTest {public static void main(String[] args) {Thread thread=new Thread(new Runnable() {@Overridepublic void run() {int x=0;while(true){String name = Thread.currentThread().getName();System.out.println("hello, ["+name+"] - "+(x++));try{Thread.sleep(1000);}catch(Exception e){e.printStackTrace();}}}},"线程的名字");  //第二个参数可以指定idthread.start();}
}

在这里插入图片描述

main线程和t线程之间的关系

在此过程中,main线程和t线程之间的关系是: main线程在执行main方法的过程中,创建并启动了t线程,并且t线程启动后,和main线程就没有关系了,这时候main线程和t线程都是自己独立的运行,并且他们俩个是要争夺CPU的时间片(使用权)的。

线程的名字

通过Thread类中的currentThread方法,可以获取当前线程的对象,然后调用线程对象的getName方法,可以获取当前线程的名字。 String name = Thread.currentThread().getName();

(注意,这里说的当前线程,指的是执行当前方法的线程,因为获取线程名字的代码肯定是写在某个方法中的,并且这个方法一定是由某个线程调用执行的。)
在这里插入图片描述

例如:

package shixun;public class myTest {public static void main(String[] args) {String name = Thread.currentThread().getName();System.out.println("执行当前main方法的线程是:"+name);Thread thread=new Thread(new Runnable() {@Overridepublic void run() {String name1 = Thread.currentThread().getName();System.out.println("执行当前run方法的线程是:"+name1);}});thread.start();}
}

在这里插入图片描述
(注意,一定要记得,start方法启动线程后,线程会自动执行run方法,千万不要直接调用run方法,这样就不是启动线程执行任务,而是普通的方法调用。)

默认情况下,主线程中,创建出的线程,它们的都会有一个默认的名字,如Thread-0 Thread-1
Thread-2等等,我们也可以创建线程对象的时候,给它设置一个指定的名字:

Thread t = new Thread("t线程");
//或者
Thread t = new Thread(new Runnable(){public void run(){//执行任务}
},"t线程");
//或者
Thread t = new Thread();
t.setName("t线程");

线程的优先级

线程的优先级使用int类型数字表示,最大是10,最小是1,默认的优先级是5。
当俩个线程争夺CPU时间片的时候:
优先级相同,获得CPU使用权的概率相同
优先级不同,那么高优先级的线程有更高的概率获取到CPU的使用权(只是有更高的概率,并不是一定能获取到)

例如:不设置优先级的情况下,t1和t2线程各自运行10000次循环,看哪个线程先运行完

package shixun;public class myTest {public static void main(String[] args) {Thread thread1=new Thread("t1线程"){@Overridepublic void run() {String name = Thread.currentThread().getName();for (int i = 0; i < 10000; i++) {}System.out.println(name+"线程执行完毕");}};Thread thread2=new Thread("t2线程"){@Overridepublic void run() {String name = Thread.currentThread().getName();for (int i = 0; i < 10000; i++) {}System.out.println(name+"线程执行完毕");}};System.out.println("t1线程的优先级:"+thread1.getPriority());System.out.println("t2线程的优先级:"+thread2.getPriority());thread1.start();thread2.start();}
}

在这里插入图片描述
(注意,默认情况下,俩个线程的优先级都是5,那个俩个线程争夺到CPU的使用权的概率一样,那么基本上俩个线程都有相同的概率先执行完10000次循环,其实t1先稍微占了那么一点点的优势,因为毕竟在主线程的代码中,先启动了t1先,然后又启动了t2线程)

然后设置优先级再来运行一下:

在这里插入图片描述

在这里插入图片描述

设置完t1和t2优先级之后,在运行结果中会明显看到优先级高的t2线程,会有更高的概率先执行完
代码。

线程状态

线程的状态分为以下几种:

线程状态描述
NEW(新建)线程刚被创建,还没调用start方法,或者刚刚调用了start方法,调用 start方法不一定"立即"改变线程状态,中间可能需要一些步骤才完成一个线程的启动。
RUNNABLE(可运行)start方法调用结束,线程由NEW变成RUNNABLE,线程存活着,并尝试抢占CPU资源,或者已经抢占到CPU资源正在运行,这俩种情况的状态都显示为RUNNABLE
BLOCKED(锁阻塞)线程A和线程B都要执行方法test,而且方法test被加了锁,线程A先拿到了锁去执行test方法,线程B这时候需要等待线程A把锁释放。这时候线程B就是处理BLOCKED
WAITING(无限期等待)一个线程在等待另一个线程执行一个(唤醒)动作时,该线程进入 Waiting状态。进入这个状态后是不能自动唤醒的,必须等待另一个线程调用notify或者notifyAll方法才能够唤醒。
TIMED_WAITING(有限期等待)和WAITING状态类似,但是有一个时间期限,时间到了,自己也会主动醒来
TERMINATED(终止)run方法执行结束的线程处于这种状态。

其实 BLOCKED,WAITING,TIMED_WAITING 这三种都属于线程阻塞,只是触发的条件不同,以及从阻塞状态中恢复过来的条件也不同而已。

线程在这三种情况的阻塞下,都具备相同的特点:
1.线程不执行代码
2.线程也不参与CPU时间片的争夺

例如:一个线程,经历的最普通的过程如下:

package shixun;public class myTest {public static void main(String[] args) {Thread thread1=new Thread("t1线程"){@Overridepublic void run() {String name = Thread.currentThread().getName();for (int i = 0; i < 100000; i++) {}}};System.out.println(thread1.getState());thread1.start();System.out.println(thread1.getState());System.out.println(thread1.getState());System.out.println(thread1.getState());System.out.println(thread1.getState());System.out.println(thread1.getState());System.out.println(thread1.getState());}
}

在这里插入图片描述

刚创建好的线程对象,就是出于NEW的状态。
线程启动后,会出于RUNNABLE状态,这个RUNNABLE状态包含俩种情况:
1.就绪状态,此时这个线程没有运行,因为没有抢到CPU的执行权
2.运行状态,此时这个线程正在运行中,因为抢到CPU的执行权

相关文章:

线程和进程

文章目录 进程和线程进程线程案例 时间片概念调度方式线程的创建和启动第一种创建方式第二种创建方式&#xff08;匿名内部类&#xff09;第三种创建方式&#xff08;Runnable接口&#xff09;main线程和t线程之间的关系 线程的名字线程的优先级线程状态 进程和线程 进程 在计…...

【JavaEE】 简单认识CPU

&#x1f435;本篇文章将对cpu的相关知识进行讲解 一、认识CPU 下图是简略的冯诺依曼体系结构图 上图中&#xff0c;存储器用来存储数据&#xff0c;注意在存储器中都是以二进制的形式存储数据的&#xff0c;CPU就是中央处理器&#xff0c;其功能主要是进行各种算术运算和各种…...

《数字图像处理-OpenCV/Python》第17章:图像的特征描述

《数字图像处理-OpenCV/Python》第17章&#xff1a;图像的特征描述 本书京东 优惠购书链接 https://item.jd.com/14098452.html 本书CSDN 独家连载专栏 https://blog.csdn.net/youcans/category_12418787.html 第17章&#xff1a;图像的特征描述 特征检测与匹配是计算机视觉的…...

考研数学什么时候开始强化?如何保证进度不掉队?

晚了。我是实在人&#xff0c;不给你胡乱吹&#xff0c;虽然晚了&#xff0c;但相信我&#xff0c;还有的救。 实话实说&#xff0c;从七月中旬考研数一复习完真的有点悬&#xff0c;需要超级高效快速... 数二的时间也有点紧张... 中间基本没有试错的时间&#xff0c;让你换…...

Node.js的下载、安装和配置

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…...

java.util.Properties类介绍

java.util.Properties 是 Java 编程语言中的一个类,用于管理应用程序的配置信息,它继承自 java.util.Hashtable 类,因此它也是基于键值对的数据结构。主要用途是存储应用程序的配置参数,比如数据库连接信息、用户设置等。 以下是 Properties 类的一些主要特点和用法: 存储…...

SpringBoot后端验证码-防止密码爆破功能

一、简介 为了防止网站的用户被通过密码典爆破。引入验证码的功能是十分有必要的。而前端的验证码又仅仅是只防君子不防小人。通过burpsuit等工具很容易就会被绕过。所以后端实现的验证码才是对用户信息安全的一大重要保障。 实现思路&#xff1a; 1.引入图形生成的依赖 2.生成…...

ChatEval:通过多代理辩论提升LLM文本评估质量

论文地址:ChatEval: Towards Better LLM-based Evaluators through Multi-Agent Debate | OpenReviewText evaluation has historically posed significant challenges, often demanding substantial labor and time cost. With the emergence of large language models (LLMs…...

关于美国服务器IP的几个常见问题

在租用美国服务器时&#xff0c;与之密切相关的一个要素就是IP&#xff0c;关于IP的问题总是有人问起&#xff0c;这里列举几项常见的问题&#xff0c;以供参考。 一、IP收费吗&#xff1f; 一般情况下&#xff0c;在租用服务器时&#xff0c;会赠送几个IP&#xff0c;因为这些…...

redis运维:sentinel模式如何查看所有从节点

1. 连接到sentinel redis-cli -h sentinel_host -p sentinel_port如&#xff1a; redis-cli -h {域名} -p 200182. 发现Redis主服务器 连接到哨兵后&#xff0c;我们可以使用SENTINEL get-master-addr-by-name命令来获取当前的Redis主服务器的地址。 SENTINEL get-master-a…...

价格疑云?格行WiFi创始人亲解谜团,性价比之王如何炼成?

随身wifi行业乱象频出&#xff0c;作为行业领跑品牌的格行随身wifi&#xff0c;关于价格问题一直备受质疑。关于设备上的“格行自有格行的骄傲”也被外界认定为是自大&#xff0c;甚至发展的线下一万多家门店也被同行不认可。近日&#xff0c;企业财经专访记者有幸采访了格行随…...

揭秘“消费即赚”的循环购模式

大家好&#xff0c;我是吴军&#xff0c;今天我将带您深入探索一种颠覆传统的新型商业模式——循环购模式。在这个模式中&#xff0c;消费者不仅能享受到购物的乐趣&#xff0c;还能通过消费获得实实在在的回报&#xff0c;甚至实现“边消费边赚钱”的奇妙体验。您是否好奇&…...

javaweb个人主页设计(html+css+js)

目录 1 前言和要求 1.1 前言 1.2 设计要求 2 预览 2.1 主页页面 2.2 个人简介 2.3 个人爱好 2.4 个人成绩有代码&#xff0c;但是图片已省略&#xff0c;可以根据自己情况添加 2.5 收藏夹 3 代码实现 3.1 主页 3.2 个人简介 3.3 个人爱好 3.4 个人成绩&#xff…...

Android常用设计模式(小白必看)

不要担心冗长&#xff0c;3分钟解决面试和学习问题&#xff0c;收藏再看 目的&#xff1a;当作一种模板&#xff0c;结合自身特点&#xff0c;针对项目需求来使用 目录 单例模式 特点&#xff1a; 实现方式&#xff1a; 1、饿汉式 2、线程安全的懒汉式 3、双重校验锁 使…...

swift获取app网络和本地网络权限

请求蓝牙权限&#xff1a; //蓝牙if #available(iOS 13.1, *) {let autostate CBManager.authorizationif(autostate .notDetermined){print("")self.manager CBCentralManager(delegate: nil, queue: DispatchQueue.main,options: [CBCentralManagerOptionShowPo…...

用LangGraph、 Ollama,构建个人的 AI Agent

如果你还记得今年的 Google I/O大会&#xff0c;你肯定注意到了他们今年发布的 Astra&#xff0c;一个人工智能体&#xff08;AI Agent&#xff09;。事实上&#xff0c;目前最新的 GPT-4o 也是个 AI Agent。 现在各大科技公司正在投入巨额资金来创建人工智能体&#xff08;AI …...

ubuntu20.04系统编译yolov8-obb.cpp代码记录

任务内容 在做ncnn-yolov8-obb模型安卓端移植的过程中&#xff0c;对开源代码进行调试。为了确认开源代码yolov8-obb.cpp可以移植开发&#xff0c;先对代码进行复现。因此在linux系统下编译yolov8-obb.cpp代码&#xff0c;验证项目中的代码是可运行的。然后再把这个代码中的模…...

JavaScript的数组与函数

数组 <script type"text/javascript">/** 知识点&#xff1a;数组* 理解&#xff1a;一维数组的容器* 概念&#xff1a;* 1.数组中的数据叫做元素* 2.元素都有编号叫做下标/索引* 3.下标从0开始* 注意&#xff1a;* 1.数组作为数据的容器…...

opencv--把cv::Mat数据转为二进制数据的保存和读取

保存 #include <opencv2/opencv.hpp> #include <iostream> #include <fstream>void saveMatToBinary(const cv::Mat& mat, const std::string& filename) {std::ofstream ofs(filename, std::ios::binary);if (!ofs.is_open()) {std::cerr <<…...

【微信小程序开发实战项目】——个人中心页面的制作

&#x1f468;‍&#x1f4bb;个人主页&#xff1a;开发者-曼亿点 &#x1f468;‍&#x1f4bb; hallo 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍&#x1f4bb; 本文由 曼亿点 原创 &#x1f468;‍&#x1f4bb; 收录于专栏&#xff1a…...

微信小程序之bind和catch

这两个呢&#xff0c;都是绑定事件用的&#xff0c;具体使用有些小区别。 官方文档&#xff1a; 事件冒泡处理不同 bind&#xff1a;绑定的事件会向上冒泡&#xff0c;即触发当前组件的事件后&#xff0c;还会继续触发父组件的相同事件。例如&#xff0c;有一个子视图绑定了b…...

多场景 OkHttpClient 管理器 - Android 网络通信解决方案

下面是一个完整的 Android 实现&#xff0c;展示如何创建和管理多个 OkHttpClient 实例&#xff0c;分别用于长连接、普通 HTTP 请求和文件下载场景。 <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas…...

【JVM】- 内存结构

引言 JVM&#xff1a;Java Virtual Machine 定义&#xff1a;Java虚拟机&#xff0c;Java二进制字节码的运行环境好处&#xff1a; 一次编写&#xff0c;到处运行自动内存管理&#xff0c;垃圾回收的功能数组下标越界检查&#xff08;会抛异常&#xff0c;不会覆盖到其他代码…...

(二)原型模式

原型的功能是将一个已经存在的对象作为源目标,其余对象都是通过这个源目标创建。发挥复制的作用就是原型模式的核心思想。 一、源型模式的定义 原型模式是指第二次创建对象可以通过复制已经存在的原型对象来实现,忽略对象创建过程中的其它细节。 📌 核心特点: 避免重复初…...

spring:实例工厂方法获取bean

spring处理使用静态工厂方法获取bean实例&#xff0c;也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下&#xff1a; 定义实例工厂类&#xff08;Java代码&#xff09;&#xff0c;定义实例工厂&#xff08;xml&#xff09;&#xff0c;定义调用实例工厂&#xff…...

相机从app启动流程

一、流程框架图 二、具体流程分析 1、得到cameralist和对应的静态信息 目录如下: 重点代码分析: 启动相机前,先要通过getCameraIdList获取camera的个数以及id,然后可以通过getCameraCharacteristics获取对应id camera的capabilities(静态信息)进行一些openCamera前的…...

【JavaWeb】Docker项目部署

引言 之前学习了Linux操作系统的常见命令&#xff0c;在Linux上安装软件&#xff0c;以及如何在Linux上部署一个单体项目&#xff0c;大多数同学都会有相同的感受&#xff0c;那就是麻烦。 核心体现在三点&#xff1a; 命令太多了&#xff0c;记不住 软件安装包名字复杂&…...

网络编程(UDP编程)

思维导图 UDP基础编程&#xff08;单播&#xff09; 1.流程图 服务器&#xff1a;短信的接收方 创建套接字 (socket)-----------------------------------------》有手机指定网络信息-----------------------------------------------》有号码绑定套接字 (bind)--------------…...

企业如何增强终端安全?

在数字化转型加速的今天&#xff0c;企业的业务运行越来越依赖于终端设备。从员工的笔记本电脑、智能手机&#xff0c;到工厂里的物联网设备、智能传感器&#xff0c;这些终端构成了企业与外部世界连接的 “神经末梢”。然而&#xff0c;随着远程办公的常态化和设备接入的爆炸式…...

大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计

随着大语言模型&#xff08;LLM&#xff09;参数规模的增长&#xff0c;推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长&#xff0c;而KV缓存的内存消耗可能高达数十GB&#xff08;例如Llama2-7B处理100K token时需50GB内存&a…...