【Java并发编程之美 | 第一篇】并发编程线程基础

在这里插入图片描述

文章目录

  • 1.并发编程线程基础
    • 1.1什么是线程和进程?
    • 1.2线程创建与运行
      • 1.2.1继承Thread类
      • 1.2.2实现Runnable接口
      • 1.2.3实现Callable接口(与线程池搭配使用)
      • 1.2.4小结
    • 1.3线程常用方法
      • 1.3.1线程等待与通知
      • 1.3.2线程睡眠
      • 1.3.3让出CPU执行权
      • 1.3.4线程中断
    • 1.4理解线程上下文切换
    • 1.5线程死锁
      • 1.5.1什么是线程死锁?
      • 1.5.2如何避免死锁?

1.并发编程线程基础

1.1什么是线程和进程?

  1. 线程是进程中的一个实体,线程本身是不会独立存在的,线程则是进程的一个执行路径
  2. 进程是系统进行资源分配的基本单位,线程是CPU分配的基本单位
  3. 进程例子:我们在电脑上启动的一个个应用,比如我们启动一个浏览器,就会启动了一个浏览器进程
  4. 线程例子:在 Java 程序中启动的一个 main 函数,即启动了一个JVM进程,而main函数所在的线程就是这个进程中的一个线程,称为主线程

image-20240614185816073

1.2线程创建与运行

  1. Java中创建线程主要有三种⽅式,分别为继承Thread类、实现Runnable接口、实现Callable接口。

1.2.1继承Thread类

  1. 继承Thread类,重写run()⽅法,调⽤start()⽅法启动线程
public class ThreadTest {/*** 继 承Thread类**/public static class MyThread extends Thread {@Overridepublic void run () {System.out.println( "This is child thread" ) ;}}public static void main ( String [] args) {MyThread thread = new MyThread ();thread.start();}
}

1.2.2实现Runnable接口

  1. 实现 Runnable 接口,重写 run() 方法
  2. 然后创建 Thread 对象,将 Runnable 对象作为参数传递给 Thread 对象,调用 start() 方法启动线程。
class RunnableTask implements Runnable {public void run() {System.out.println("上岸、上岸!");}public static void main(String[] args) {RunnableTask task = new RunnableTask();Thread thread = new Thread(task);thread.start();}
}

1.2.3实现Callable接口(与线程池搭配使用)

  1. 实现 Callable 接口,重写 call() 方法
  2. 然后创建 FutureTask 对象,参数为 Callable 对象;紧接着创建 Thread 对象,参数为 FutureTask 对象,调用 start() 方法启动线程。
  3. 通过 实现Callable接口的对象 的get方法获取返回结果
class CallableTask implements Callable<String> {public String call() {return "上岸、上岸了!";}public static void main(String[] args) throws ExecutionException, InterruptedException {CallableTask task = new CallableTask();FutureTask<String> futureTask = new FutureTask<>(task);Thread thread = new Thread(futureTask);thread.start();System.out.println(futureTask.get());}
}

1.2.4小结

image-20240421103020296

1.3线程常用方法

  1. 线程等待方法:wait()、wait(long timeout)、wait(long timeout,int nanos)
  2. 线程通知方法:nodify()、notifyAll()
  3. 让出优先权:yield()
  4. 线程中断方法:interrupt()、isinterrupted()、interrupted()
  5. 线程休眠方法:sleep()

1.3.1线程等待与通知

线程等待方法:

  1. wait():当一个线程 A 调用一个共享变量的 wait() 方法时,线程 A 会被阻塞挂起,直到发生下面几种情况才会返回 :

    1.1 线程 B 调用了共享对象 notify()或者 notifyAll() 方法;

    1.2 其他线程调用了线程 A 的 interrupt()方法,线程 A 抛出 InterruptedException 异常返回。

  2. wait(long timeout) :这个方法相比 wait() 方法多了一个超时参数,它的不同之处在于,如果线程 A 调用共享对象的 wait(long timeout)方法后,没有在指定的 timeout 时间内被其它线程唤醒,那么这个方法还是会因为超时而返回。、

  3. wait(long timeout, int nanos),其内部调用的是 wait(long timout) 方法。

唤醒/通知线程主要有下面两个方法:

  1. notify():一个线程 A 调用共享对象的 notify() 方法后,会唤醒一个在这个共享变量上调用 wait 系列方法后被挂起的线程。一个共享变量上可能会有多个线程在等待,具体唤醒哪个等待的线程是随机的
  2. notifyAll():不同于在共享变量上调用 notify()方法会唤醒被阻塞到该共享变量上的一个线程,notifyAll 方法会唤醒所有在该共享变量上调用 wait 系列方法而被挂起的线程。

join():等待线程执行终止

  1. 如果一个线程 A 执行了 thread.join(),当前线程 A 会被阻塞,即等待 thread 线程执行终止之后才从 thread.join() 返回

1.3.2线程睡眠

  1. sleep(long millis):Thread 类中的静态方法,当一个执行中的线程 A 调用了 Thread 的 sleep 方法后,线程 A 会暂时让出指定时间的执行权
  2. 但是线程 A 所拥有的监视器资源,比如锁,还是持有不让出的。指定的睡眠时间到了后该方法会正常返回,接着参与 CPU 的调度,获取到 CPU 资源后就可以继续运行

1.3.3让出CPU执行权

  1. yield():Thread 类中的静态方法,当一个线程调用 yield 方法时,实际是在暗示线程调度器,当前线程请求让出自己的 CPU

1.3.4线程中断

  1. Java 中的线程中断是一种线程间的协作模式,通过设置线程的中断标志并不能直接终止该线程的执行

  2. void interrupt():中断线程

    2.1 例如:当线程A运行时,线程B可以调用线程A的interrupt()方法来设置线程A的中断标志为true并立即返回

    2.2 设置中断标志仅仅是标记,线程A并没有被中断,会继续往下执行

    2.3 但如果线程A因为调用wait、join、以及sleep方法而被阻塞挂起,这时线程B若调用线程A的interrupt()方法,线程A会在调用这些方法的地方抛出InterrupedException异常

  3. boolean isInterrupted():检测当前线程是否被中断,如果是返回true,否则返回false

  4. boolean interrupted():检测当前线程是否被中断,如果是返回true,否则返回false。与 isInterrupted 不同的是,该方法如果发现当前线程被中断,则会清除中断标志。

1.4理解线程上下文切换

  1. 在多线程编程中,线程个数一般都大于CPU个数,但是每个CPU同一时刻只能被一个线程使用
  2. 为了让用户感觉多个线程是在同时执行,CPU资源的分配采用了时间片轮转的方法,即给每个线程分配一个时间片,线程在时间片内占用CPU执行任务。当线程使用完时间片,就处于就绪状态并让出CPU让其他线程占用,即上下文切换

1.5线程死锁

1.5.1什么是线程死锁?

  1. 死锁是指两个或两个以上的线程在执行过程中,因为争夺资源而造成的互相等待的现象

  2. 产生死锁的四个条件:

    2.1 互斥性:资源是互斥的,同一时刻只能由一个线程占用

    2.2 请求并持有条件:一个线程已经占有一个资源,同时提出新的资源请求,并占据已有的资源不释放

    2.3 不可剥夺条件:线程获取到的资源在自己使用完之前不能被其他线程所占用

    2.4 环路等待条件:指在发生死锁时,必然存在一个线程一资源的环形链

  3. 代码举例:

    import java.util.Date;public class LockTest {public static String obj1 = "obj1";public static String obj2 = "obj2";public static void main(String[] args) {LockA la = new LockA();new Thread(la).start();LockB lb = new LockB();new Thread(lb).start();}
    }
    class LockA implements Runnable{public void run() {try {System.out.println(new Date().toString() + " LockA 开始执行");while(true){synchronized (LockTest.obj1) {System.out.println(new Date().toString() + " LockA 锁住 obj1");Thread.sleep(3000); // 此处等待是给B能锁住机会synchronized (LockTest.obj2) {System.out.println(new Date().toString() + " LockA 锁住 obj2");Thread.sleep(60 * 1000); // 为测试,占用了就不放}}}} catch (Exception e) {e.printStackTrace();}}
    }
    class LockB implements Runnable{public void run() {try {System.out.println(new Date().toString() + " LockB 开始执行");while(true){synchronized (LockTest.obj2) {System.out.println(new Date().toString() + " LockB 锁住 obj2");Thread.sleep(3000); // 此处等待是给A能锁住机会synchronized (LockTest.obj1) {System.out.println(new Date().toString() + " LockB 锁住 obj1");Thread.sleep(60 * 1000); // 为测试,占用了就不放}}}} catch (Exception e) {e.printStackTrace();}}
    }
    
  4. 执行结果:

    image-20240614201836475

1.5.2如何避免死锁?

  1. 只要破坏产生死锁的四大条件中的一个即可,其中破坏环形等待条件最为容易,即保持资源申请的有序性就可以避免死锁

  2. 例子:

    class LockB implements Runnable{public void run() {try {System.out.println(new Date().toString() + " LockB 开始执行");while(true){synchronized (LockTest.obj1) {System.out.println(new Date().toString() + " LockB 锁住 obj2");Thread.sleep(3000); synchronized (LockTest.obj2) {System.out.println(new Date().toString() + " LockB 锁住 obj1");Thread.sleep(60 * 1000); }}}} catch (Exception e) {e.printStackTrace();}}
    }
    

在这里插入图片描述

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

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

相关文章

漫步者开放式耳机值得买吗?漫步者、西圣、小米硬核测评pk性能!

说起开放式蓝牙耳机&#xff0c;相信大部分朋友都不会陌生。与传统的封闭式耳机相比&#xff0c;开放式蓝牙耳机不仅提升了佩戴舒适度&#xff0c;还对耳朵有良好的保护效果。特别适合喜欢户外运动和长途旅行的用户。然而&#xff0c;由于市场上产品众多&#xff0c;选择一款高…

用Python向Word文档添加页眉和页脚

用Python向Word文档添加页眉和页脚 添加页眉和页脚效果代码 添加页眉和页脚 在本文中&#xff0c;我们将用python向文档中添加页眉和页脚。 效果 添加前的文档&#xff1a; 添加页眉和页脚后&#xff1a; 代码 from docx import Documentdef add_header_footer(doc_path…

UE4_后期_ben_模糊和锐化滤镜

学习笔记&#xff0c;不喜勿喷&#xff0c;侵权立删&#xff0c;祝愿生活越来越好&#xff01; 本篇教程主要介绍后期处理的简单模糊和锐化滤镜效果&#xff0c;学习之前首先要回顾下上节课介绍的屏幕扭曲效果&#xff1a; 这是全屏效果&#xff0c;然后又介绍了几种蒙版&#…

MySQL与PostgreSQL关键对比四(关联查询性能)

引言&#xff1a;MySQL单表的数据规模一般建议在百万级别&#xff0c;而PostgreSQL的单表级别一般可以到亿级&#xff0c;如果是MPP版本就会更多。从基础数据建议上&#xff0c;不难看出&#xff0c;MySQL在Join的情况下也就是主要查询的情况下性能和PostgreSQL相差还是很大的。…

Vue35-生命周期小结

一、8个&#xff0c;4对生命周期函数 第一对&#xff1a;数据监测、数据代理&#xff0c;创建之前和创建之后。 注意&#xff1a;不是vm的创建&#xff01;&#xff01;&#xff01; 第二队&#xff1a;beforeMount和mounted 第三队&#xff1a;beforeUpdate和update 第四队…

vue-element-admin后台集成方案

官网&#xff1a;介绍 | vue-element-adminA magical vue adminhttps://panjiachen.github.io/vue-element-admin-site/zh/guide 1.git环境安装配置及简单操作 1.1git环境安装配置 git软件官网&#xff1a;Git - Downloads (git-scm.com)https://git-scm.com/downloads 下载…

印尼slot游戏pwa做Facebook广告代投推广流程

印尼slot游戏pwa做Facebook广告代投推广流程 随着互联网的普及和移动设备的广泛应用&#xff0c;社交媒体平台如Facebook成为了企业和个人进行品牌推广的重要渠道。在印尼的slot游戏行业中&#xff0c;利用PWA&#xff08;Progressive Web App&#xff09;技术进行Facebook广告…

第N4周:中文文本分类-Pytorch实现

&#x1f368; 本文为&#x1f517;365天深度学习训练营 中的学习记录博客&#x1f356; 原作者&#xff1a;K同学啊 | 接辅导、项目定制&#x1f680; 文章来源&#xff1a;K同学的学习圈子 目录 一、准备工作 1.任务说明 文本分类流程图&#xff1a; 2.加载数据 ​编辑 二、…

conda添加镜像源与channels

文章目录 一、conda下添加国内镜像源&#xff08;window下&#xff09;二、pip配置国内镜像源&#xff08;window下&#xff0c;临时修改&#xff09;三、conda源的定义 一、conda下添加国内镜像源&#xff08;window下&#xff09; 1、为【channels】配置清华镜像通道 直接在…

【Pandas】已完美解决:AttributeError: ‘DataFrame‘ object has no attribute ‘ix‘

文章目录 一、问题背景二、可能出错的原因三、错误代码示例四、正确代码示例&#xff08;结合实战场景&#xff09;五、注意事项 一、问题背景 在Pandas的早期版本中&#xff0c;ix 是一个方便的索引器&#xff0c;允许用户通过标签和整数位置来索引DataFrame的行和列。然而&a…

Pixi.js学习 (四)鼠标跟随、元素组合与图片位控

目录 一、鼠标移动跟随 1.1 获取鼠标坐标 1.2 鼠标跟随 二、锚点、元素组合 2.1 锚点 2.2 元素组合 三、图片图层 四、实战 例题一&#xff1a;完成合金弹头人物交互 例题二&#xff1a;反恐重击瞄准和弹痕 例题一代码&#xff1a; 例题二代码&#xff1a; 总结 前言 为了提高作…

ADS基础教程20 - 电磁仿真(EM)参数化

EM介绍 一、引言二、参数化设置1.参数定义2.参数赋值3.创建EM模型和符号 四、总结 一、引言 参数化EM仿真&#xff0c;是在Layout环境下创建参数&#xff0c;相当于在原理图中声明变量。 二、参数化设置 1.参数定义 1&#xff09;在Layout视图&#xff0c;菜单栏中选中EM&g…

大模型出现的不断重复的现象

无论是大语言模型还是多模态模型,都遇到过这个问题,该如何解决呢? 1.调整推理参数 [BUG] 返回重复的内容 Issue #277 QwenLM/Qwen GitHub是否已有关于该错误的issue或讨论? | Is there an existing issue / discussion for this? 我已经搜索过已有的issues和讨论 | I…

【Linux】基础IO——系统文件IO

我之前是讲过c语言的文件操作的&#xff0c;但是说实话我压根就不知道它在干什么&#xff0c;后面c语言/c,数据结构的学习过程中也没用过文件操作&#xff0c;今天我们就来会会这个文件操作 1.回顾c语言文件接口 1.1.fopen r &#xff1a;只读模式打开&#xff0c;文件流指针…

【LeetCode 92.】 反转链表 II

1.题目 虽然本题很好拆解&#xff0c;但是实现起来还是有一些难度的。 2. 分析 尽可能抽象问题&#xff0c;然后简化代码 我在写本题的时候&#xff0c;遇到了下面这两个问题&#xff1a; 没有把[left,right] 这个区间的链表给断开&#xff0c;所以导致反转起来非常麻烦。…

【iOS】KVO相关总结

目录 1. 什么是KVO&#xff1f;2. KVO的基本使用3. KVO的进阶使用observationInfo属性context 的使用KVO触发监听方法的方式自动触发手动触发 KVO新旧值相等时不触发KVO的从属关系一对一关系一对多关系 4. KVO使用注意5. KVO本质原理分析伪代码保留伪代码下的类并编译运行对比添…

小白都能看懂的 “栈”

什么是栈&#xff1f;首先引用维基百科的解释&#xff1a; 栈&#xff08;stack&#xff09;是计算机科学中的一种抽象资料类型&#xff0c;只允许在有序的线性资料集合的一端&#xff08;称为堆栈顶端&#xff0c;top&#xff09;进行加入数据&#xff08;push&#xff09;和移…

Go语言结构体内嵌接口

前言 在golang中&#xff0c;结构体内嵌结构体&#xff0c;接口内嵌接口都很常见&#xff0c;但是结构体内嵌接口很少见。它是做什么用的呢&#xff1f; 当我们需要重写实现了某个接口的结构体的(该接口)的部分方法&#xff0c;可以使用结构体内嵌接口。 作用 继承赋值给接口…

信号与系统实验MATLAB-实验1-信号的MATLAB表示及信号运算

实验1-信号的MATLAB表示及信号运算 一、实验目的 1、掌握MATLAB的使用&#xff1b; 2、掌握MATLAB生成信号波形&#xff1b; 3、掌握MATLAB分析常用连续信号&#xff1b; 4、掌握信号运算的MATLAB实现。 二、实验内容 编写程序实现下列常用函数&#xff0c;并显示波形。…

PyTorch -- Visdom 快速实践

安装&#xff1a;pip install visdom 注&#xff1a;如果安装后启动报错可能是 visdom 版本选择问题 启动&#xff1a;python -m visdom.server 之后打开出现的链接 http://localhost:8097Checking for scripts. Its Alive! INFO:root:Application Started INFO:root:Working…