多线程的学习上篇

news2024/11/23 0:55:42

previewfile_2862005448123

座右铭: 天行健,君子以自强不息;地势坤,君子以厚德载物.

引入进程这个概念的目的

引入进程这个概念,最主要的目的,是为了解决“并发编程"这样的问题.
这是因为CPU进入了多核心的时代
要想进一步提高程序的执行速度,就需要充分的利用CPU 的多核资源

其实,多进程编程,已经可以解决并发编程的问题了.已经可以利用起来CPU多核资源了.
但是它带来了新的问题 => 那就是进程太重了(消耗资源多&速度慢)!!!
创建一个进程,开销比较大.
销毁一个进程,开销也比较大.
调度一个进程,开销还比较大.

进程虽然能解决问题,但是它不是作为这个问题的最优解法
期望一种开销更小,重量更轻的这样的一个概念来解决这个“并发编程”
因此,线程应运而生!!!线程也叫做“轻量级进程”
它在解决“并发编程”问题的前提下,让创建,销毁,调度的数速度,更快了一些!!!

进程是操作系统中的基本单位.创建进程,就得给它分配一些资源(如内存资源和硬盘资源……)
在分配资源的时候,还得消耗一些时间进程重,主要就是重在"资源分配/回收"上.

为什么线程更轻量???

=> 它把申请资源/释放资源的操作给省下了.

以工厂的扩建为例:
方案一: 在另外的地方,再建一个跟原来工厂一模一样的分厂(工厂二号)
~~ 需要新的场地和新的物流体系,一套新的生产线….
=> 多进程版本的方案

image-20230918093634503

方案二: 在原来工厂上进行扩建,增加一套生产线 ~~ 成本比第一种方案小很多
~~ 场地和物流体系,都是可以复用之前的,只需要一套新的生产线……
~~ 多线程版本的方案 => 此时,只是在第一套生产线的时候,才需要把资源申请到位,
后续再加新的生产线,就复用之前的资源即可.

image-20230918094220221

线程和进程的关系

~~ 是进程包含线程
一个进程可以包含一个线程,也可以包含多个线程.(不能没有)
只有第一个线程启动的时候,开销是比较大的.后续线程就省事了.

同一个进程里的多个线程之间,共用了进程的同一份资源(主要指的是内存文件描述符表)

  • 内存 ~~ 线程1new的对象,在线程2,3,4里都可以直接使用
  • 文件描述表 ~~线程1打开的文件,在线程2,3,4里都可以直接使用

操作系统,实际调度的时候,是以线程为单位进行调度的.
(之前那篇博客讲进程调度,相当于每个进程里面只有一个线程这样的情况)

如果每个进程有多个线程了,每个线程是独立在CPU上调度的
=> 线程是操作系统调度执行的基本单位

一个线程可以在多个进程里面嘛?
不可以!!进程和线程是一个—对多的关系.

一个核心上执行的是一个线程.如果你一个进程有两个线程.
线程1可能在核心A上执行,线程2可能在核心B上执行.
操作系统调度的时候,其实不关心进程,而是只关心线程

一个线程也是通过一个PCB 来描述的
一个进程里面可能是对应一个PCB,也可能是对应多个PCB
之前介绍的,PCB里的状态,上下文,优先级,记账信息,都是每个线程有自己的.各自记录各自的.
但是同一个进程里的PCB之间,pid是一样的,内存指针和文件表述表也是一样的
=> 根据相同的pid来区分他们是同一个进程的.

谈到"调度”已经和进程没啥关系了.
进程专门负责资源分配.
线程来接管和调度相关的一切内容.

理论是为了指导我们编程的,不懂这个理论,
就不知道这个代码为什么这么写!!!

图示理解

最初的需求
1个滑稽老铁,吃100只鸡 ~~ 效率比较低
image-20230918154821678

如何吃得快点,有多进程多线程两个版本,如下图:
多进程吃鸡的方式:
~~ 成本比较高

image-20230918155101436

多线程吃鸡的方式:

image-20230918155347489提高这里滑稽老铁的个数,速度能否更快了???

image-20230918160353714

但是如果继续提高滑稽老铁的个数了???
image-20230918163513486

(1)在多线程的情况下,多个滑稽老铁,共享同一份鸡肉,此时就可能会打架

image-20230918170001294

(2)多线程还有一个情况 ~~

image-20230918163927661

注: chrome浏览器要使用多进程编程模型(每个标签页都是个进程)
~~ 目的就是为了防止,一个页面挂了把别的页面带走.

什么时候会存在安全问题???

多个执行流访问同一个共享资源的时候.
~~ 线程模型,天然就是资源共享的.多线程争抢同一个资源(同一个变量)非常容易触发的.
~~ 进程模型,天然是资源隔离的,不容易触发.
进行进程间通信的时候,多个进程访问同一个资源,可能会出问题.

多线程会提高效率,但是不如多进程安全.
但是代码写的好,线程安全问题,就不怕
=> 当前多线程的编程模型,要比多进程这种模型应用更加广泛.
工作中遇到多线程的情况也要比多进程多很多.尤其是Java圈子.

面试中,进程和线程的区别,就是"操作系统"这个模块最高频的问题,没有之一!!!




在Java中如何进行多线程编程

本身关于线程的操作,都是操作系统提供的API
Java是个跨平台的语言.很多操作系统的提供的功能,都被JVM给封装好了.
咱们不需要学习系统原生API,只需要学习Java提供的API就行了.


Java是如何实现跨平台的???
  • windows系统,实现了一个windows版本的JVM

  • Linux系统,实现了Linux版本的JVM

  • Mac系统,实现了Mac 版本的JVM

  • 随便某个系统, 也实现了对应版本的JVM

跨平台是靠无数个不同版本的JVM支持的!
这些不同的JVM内部封装了不同系统的API,对上都是执行同样规则的字节码
例子: 你去欧洲周游,到各个国家打卡
到了英国,就去请一个英国的翻译(既会说英语,也会说汉语),帮助我与本地人沟通;
到了法国,就又找一个法国的翻译(既会说法语,也会说汉语),帮助我与本地人沟通;
到了德国,就又找一个德国的翻译(既会说德语,也会说汉语),帮助我与本地人沟通;
………
Java跨平台,就于有几个平台,就有几个“翻译”…
~~ 点评: “大力出奇迹!!!”


Java操作多线程,最核心的类Thread

~~ 使用Thread类,不需要import别的包,因为它在java.lang下面的
类似的有: String, StringBuilder, StringBuffer

创建线程,是希望线程成为一个独立的执行流,它要能够执行一段代码.

第一个多线程程序

代码如下:

class MyThread extends Thread{
    @Override
    public void run() {
        while(true){
            System.out.println("hello thread");
            // 为了让这里的打印慢点, 方便看, 加个sleep, 休眠 1s
            try {
                Thread.sleep(1000);
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }
    }
}

public class ThreadDemo1 {
    public static void main(String[] args) {
        Thread t = new MyThread();
        t.start();

        while(true){
            System.out.println("hello main");
            try {
                Thread.sleep(1000);
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }
    }
}

运行结果:

image-20230918212013737

操作系统调度线程的时候,“抢占式执行”,具体哪个线程先上,哪个线程后上,不确定,取决于操作系统调度器具体实现策略.
虽然有优先级,但是在应用程序层面上无法修改.
从应用程序(代码)的角度,看到的效果,就好像是线程之间的调度顺序是"随机"的一样.
=> 内核里本身并非是随机.但是干预因素太多,并且应用程序这一层也无法感知到细节,
就只能认为是随机的了!!!
执行顺序是由内核实现的,无解,但可以通过一些API进行有限度的干预.
线程安全问题的罪魁祸首,万恶之源,就是这里的抢占式执行,随机调度.

这里的Thread类,本质上还是系统里的线程的封装.
每个Thread的对象就对应到系统中的—个线程(也就对应一个PCB).

方法重写 Override: 父类有个方法,子类搞一个名字相同,参数也完全相同方法。
子类的方法就会被动态绑定的机制,被调用到.
Thead t = new MyThread();
t.run(); 虽然t是父类引用,此处调用的run仍然是子类的方法.(t本质上还是指向的子类的对象)

重载 Overload: 同一个作用域,多个方法,名字相同,参数个数或者类型不同,
同一个类里,或者是子类和父类之间…

t.sart(); => 线程中的特殊方法,启动一个线程.
start 里面没有调用run,而是创建了个线程.由新的线程来执行run方法.
start如何创建新的线程 => 就是调用操作系统的API,
通过操作系统内核创建新线程的 PCB(程序控制块),并且把要执行的指令交给这个PCB.
当 PCB 被调度到CPU上执行的时候,也就执行到了线程run方法中的代码了
如果run方法执行完毕,新的这个线程自然销毁.

new Thread对象这个操作不创建线程.(这里说的线程指的是系统内核里的PCB)
调用start才是创建的PCB,这才有货真价实的线程的.

start 和 run之间的区别:
start是真正创建了一个线程(从系统这里创建的),线程是独立的执行流.
run 只是描述了线程要干的活是啥.如果直接再main中调用run, 此时没有创建新线程,全是main线程一个人干活.

PCB对应的是线程.一个线程对应一个PCB一个进程对应多个PCB.
如果一个进程只有一个线程,就是一个进程对一个PCB了.
同一个进程里的若干PCB pid相同.不同进程的pid是不同的.
PCB不是"简称",是一个数据结构,体现的是进程/线程是如何实现,如何被描述出来的.
PCB只是操作系统书里说的概念.实际上Linux对应结构体名字叫做task_struct

如何直观的看到上述代码中的两个线程???
可以使用JDK自带的工具 jconsole 查看当前的Java进程中的所有线程.
image-20230918215407492

注: JDK => Java开发工具包,这里带的工具很多的,不只是javac和java.
image-20230918220407913

注:进程包含线程 => 要想看到线程,得先找到对应的进程,再看进程里有哪些线程.

点击一下上图中的thread.ThreadDemo1
image-20230918220859922

image-20230918221747327

如果你打开JConsole,在本地进程什么都看不到,尝试右键,以管理员方式运行.

image-20230918224445511



Java中创建线程的多种写法
  1. 继承Thread,重写run方法

    1.继承 Thread 来创建一个线程类
    2.创建 MyThread 类的实例
    3.调用 start 方法启动线程

    
    class MyThread extends Thread{
        @Override
        public void run() {
            System.out.println("hello thread");
        }
    }
    public class ThreadDemo1 {
        public static void main(String[] args) {
            Thread t = new MyThread();
            t.start();
        }
    }
    
  2. 实现Runnable 接口

    1.实现 Runnable 接口
    2.创建 Thread 类实例, 调用 Thread 的构造方法时将 Runnable 对象作为 target 参数
    3.调用 start 方法

    Runnable 作用,是描述一个"要执行的任务",run方法就是任务的执行细节

    class MyRunnable implements Runnable {
        @Override
        public void run() {
            System.out.println("hello thread");
        }
    }
    public class ThreadDemo2 {
        public static void main(String[] args) {
            // 这只是描述了一个任务
            Runnable runnable = new MyRunnable();
            // 把任务交给线程来执行
            Thread t = new Thread(runnable);
            t.start();
        }
    }
    

    这样写的好处:
    解耦合.目的就是为了让线程和和线程要干的活之间分离开.
    未来如果要改代码,不用多线程,使用多进程,或者线程池,或者协程…此时代码改动比较小.

  3. 使用匿名内部类, 继承 Thread

    1.创建了一个Thread的子类.(子类没有名字)所以才叫做“匿名”
    2.创建了子类的实例,并且让t引用指向该实例

    public class ThreadDemo3 {
        public static void main(String[] args) {
            Thread t = new Thread(){
                @Override
                public void run() {
                    System.out.println("hello");
                }
            };
            t.start();
        }
    }
    
  4. 使用匿名内部类, 实现 Runnable

    这个写法和方法2本质相同.
    只不过是把实现Runnable任务交给匿名内部类的语法.
    此处是创建了一个类,实现 Runnable,同时创建了类的实例,并且传给Thread的构造方法.

    public class ThreadDemo4 {
        public static void main(String[] args) {
            Thread t =new Thread(new Runnable() {
                @Override
                public void run() {
                    System.out.println("hello");
                }
            });
            t.start();
        }
    }
    
  5. 使用Lambda表达式 ~~ 最简单,推荐使用的写法

    把任务用lambda表达式来描述
    直接把 lambda传给Thread构造方法.
    lambda就是个匿名函数(没有名字的函数)用一次就销毁了.
    Java里面函数没法脱离类存在.java为了能和别的语言对齐, 搞了个蹩脚的函数式接口,通过这个来实现lambda .

    public class ThreadDemo5 {
        public static void main(String[] args) {
            Thread t = new Thread(()-> {
                System.out.println("hello");
            });
            t.start();
        }
    }
    

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1021268.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

部署Envoy Sidecar代理:演示如何将Envoy作为Sidecar代理注入到应用容器中

🌷🍁 博主猫头虎 带您 Go to New World.✨🍁 🦄 博客首页——猫头虎的博客🎐 🐳《面试题大全专栏》 文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺 &a…

【pytest】 pytest拓展功能 PermissionError问题

目录 1. pytest-html 1.1 PermissionError: [Errno 13] Permission denied: D:\\software\\python3\\anconda3\\Lib\\site-packages\\pytest_html\\__pycache__\\tmp_ttoasov 1.2错误原因 2. 失败用例重试 3. 用例并行执行 pytest-parallel 1. pytest-html 管理员打开 A…

使用postman测试邮件接口

首先找到token的位置 找到token的值之后 复制接口文档里的地址,在接口文档给的底之前前加api 配置token 在params参数里增加token参数 值复制浏览器里的token参数 发送send就ok了

交换机的工作原理(含实例,华为ensp操作)

目录​​​​​​​ ​​​​​​​1.交换机学习和转发 案例 1.设置静态地址表项 2.配置黑洞mac地址表项 1.交换机学习和转发 交换机工作在数据链路层。当交换机从某个端口收到一个帧时,它并不是向所有的接口转发此帧,而是根据此帧的目的MAC地址&a…

JMeter:接口测试基础介绍

一、什么是接口 接口是非常抽象的概念,先来看下中国最大的综合性辞典《辞海》是怎样定义接口的: 两个不同系统或系统中两个不同特性部分的交接部分。一般分硬件接口和软件接口两种。前者是为连接计算机各部分之间、计算机与计算机之间、计算机与外部系统…

Matlab论文插图绘制模板第114期—带图形标记的图

之前的文章中,分享了Matlab带线标记的图: 带阴影标记的图: 带箭头标记的图: 进一步,分享一下带图形标记的图,先来看一下成品效果: 特别提示:本期内容『数据代码』已上传资源群中&…

9.18 QT作业

mainwindow.h QT_BEGIN_NAMESPACE namespace Ui { class MainWindow; } QT_END_NAMESPACEclass MainWindow : public QMainWindow {Q_OBJECTpublic:MainWindow(QWidget *parent nullptr);~MainWindow();signals:void jump(); //自定义跳转信号函数private slots:vo…

小插曲 -- 使用Linux编写 判断程序是否在运行的小程序

编写思路 首先,在执行“ps -elf |grep xxx”时,如果xxx存在,通常会有两条结果,一个是xxx对应的PID,一个则是grep对应的PID,但是如果我希望执行命令后,xxx存在就只有xxx对应的PID,不…

Vue3函数式编程

文章目录 前言一、三种编程风格1.template2.jsx/tsx3.函数式编写风格 二、函数式编程1.使用场景2.参数3.例子3.render渲染函数 总结 前言 本文主要记录vue3中的函数式编程以及其他编程风格的简介 一、三种编程风格 1.template Vue 使用一种基于 HTML 的模板语法,…

QML元素定位器:Row、Colum、Grid、Flow、定位器嵌套以及Repeater用法

在QML中,定位器(Positioner)是一种特殊的组件,用于管理其子项的位置。定位器确保其子项始终根据给定的规则排列。Qt Quick提供了几种定位器,用于处理不同的布局需求。 以下是一些常用的QML定位器: Row:将其子项沿水平线排列。Column:将其子项沿垂直线排列。Grid:将其子…

6.3 字符数组

思维导图: 前言: 主要内容: 前言内容整理 字符型数据和存储 字符型数据是依据字符的ASCII代码存储在内存单元中,通常占用一个字节的空间。ASCII代码可以被认为是整数,因此在C99标准中,字符类型被归类为整…

若依DataScopeAspect数据权限解析和ew.customSqlSegment源码解析

目录 一、DataScopeAspect使用场景二、ew.customSqlSegment${ew.customSqlSegment}build:this.normal : queryWrapper where 条件不为空的时候,才有normalget第二次 进来add(), 已经拼接完 ew.customSqlSegment 了, 因为DataPermission 注解进…

开发一个训练LORA的WebUI

文章目录 效果原理说明代码 效果 原理 基于开源项目kohya-ss/sd-scripts增加了Gradio UI,精简了训练参数,更适合初级“炼丹宝宝”的炼丹炉! 核心思想还是比较简单的,通过Gradio UI来收集设置的训练参数,并通过python的…

全国职业技能大赛云计算--高职组赛题卷⑤(私有云)

全国职业技能大赛云计算--高职组赛题卷⑤(私有云) 第一场次题目:OpenStack平台部署与运维任务1 基础运维任务(5分)任务2 OpenStack搭建任务(15分)任务3 OpenStack云平台运维(15分&am…

渗透测试信息收集方法

一、域名收集 OneForAll输出表格方便筛选(status、title) layer5.0saintsec更新版子域名挖掘机 百度云链接: https://pan.baidu.com/s/1VQ2HLocs39B72ElysskPog 提取码:121l subdomainsBurte,python2子域名爆破 https://github.com/y1ng1996/lijiejie_subDomainsBru…

偶现来电时手机操作出现重启

问题描述:偶现来电时手机操作出现重启 问题分析:从系统Log看 09-06 10:22:44.791829 1400 1425 W Watchdog: *** WATCHDOG KILLING SYSTEM PROCESS: Blocked in handler on main thread (main) 09-06 10:22:44.794133 1400 1425 W Watchdog: main …

Python Web开发:构建动态Web应用

💂 个人网站:【工具大全】【游戏大全】【神级源码资源网】🤟 前端学习课程:👉【28个案例趣学前端】【400个JS面试题】💅 寻找学习交流、摸鱼划水的小伙伴,请点击【摸鱼学习交流群】 Python已经成为一门流行…

功能基础篇2——常用哈希和加密算法介绍及Python相关库与实现

加解密 https://docs.python.org/3/library/crypto.html 三方库推荐,https://cryptography.io/en/latest/ Criptography,https://pypi.org/project/cryptography/ PyCryptodome,a fork of PyCrypto,https://pypi.org/project/…

vue3+ts 实现移动端分页

current 开始页码 pageSize 结束页码 const sizeref<number>(10) //一页显示十条 const eachCurrentPageref<number>(1) //默认是第一页interface ITdata {current: number,pageSize: number,// xxxx 其他参数... } const selectApplyList ref<…

vue-h5:移动Web单击事件和延迟300ms的问题

在PC端的网页&#xff0c;大部分的交互是通过click事件来实现的&#xff0c;然而在移动端&#xff0c;则是通过touch事件来实现触摸交互。 单击或者点击事件&#xff0c;指的是鼠标按下并且在短时间内放开【一般是小于300ms】。 那么移动端&#xff0c;也是类似&#xff0c;在…