【多线程】Java如何实现多线程?如何保证线程安全?如何自定义线程池?

在这里插入图片描述

个人简介:Java领域新星创作者;阿里云技术博主、星级博主、专家博主;正在Java学习的路上摸爬滚打,记录学习的过程~
个人主页:.29.的博客
学习社区:进去逛一逛~

在这里插入图片描述

多线程

  • Java多线程
    • 1. 进程与线程
    • 2. 多线程
      • 1) 相关概念
      • 2) 多线程实现方式
        • ①继承Thread类
        • ②实现Runnable接口
        • ③利用Callable接口和Future接口
        • ④ 比较优缺点
    • 3. Thread类 常用方法
    • 4. 线程安全
      • 1) 同步代码块
      • 2) 同步方法
      • 3) Lock锁
    • 5. 等待唤醒机制(生产者和消费者)
      • 1) 生产者
      • 2) 消费者
      • 测试:
      • 3) 等待唤醒 —— 阻塞队列方式
    • 6. 线程状态
    • 7. 线程池
      • 自定义线程池


Java多线程


1. 进程与线程

线程

  • 线程是操作系统能够进行运算调度的最小单位。它被包含在进程中,是进程中的实际运作单位。

  • 生命周期:

  • 在这里插入图片描述


进程

  • 进程是程序的基本执行实体。



2. 多线程


1) 相关概念

相关概念

  • 并发:在同一时刻,有多个指令在单个CPU上交替执行。
  • 并行:在同一时刻,有多个指令在单个CPU上同时进行。

2) 多线程实现方式

①继承Thread类

多线程第一种实现方式

  • ①继承Thread类

  • ②重写run方法

  • ③创建子类的对象,并使用start()方法启动线程

  • /*** @author .29.* @create 2023-10-17 16:21*/
    public class extendThread {//1.自定义类,继承Thread类static class myThread extends Thread{//2.重写run方法,编写执行逻辑@Overridepublic void run() {for(int i = 0;i < 100; ++i){System.out.println("执行:" + getName());}}}public static void main(String[] args){//3. 实现Thread子类对象,使用start()启动线程myThread t1 = new myThread();myThread t2 = new myThread();t1.setName("线程1号");t2.setName("线程2号");t1.start();t2.start();}
    }
    
②实现Runnable接口

多线程第二种实现方式

  • ①自定义类,实现Runnable接口。
  • ②重写接口中的run方法。
  • ③实例化Runnable实现类
  • ④以Runnable实现类对象为参数,创建Thread实例,开启线程。
public class implementsRunnable {//1.自定义类,实现Runnable接口static class myRun implements Runnable{//2.重写抽象方法@Overridepublic void run() {//编写需要执行的程序for(int i = 0;i < 100; ++i){//获取当前执行的线程Thread t = Thread.currentThread();System.out.println("执行:" + t.getName());}}}public static void main(String[] args){//3.实例化Runnable实现类myRun mr = new myRun();//4.创建线程对象,使用start()启动线程Thread t1 = new Thread(mr);Thread t2 = new Thread(mr);t1.start();t2.start();}
}
③利用Callable接口和Future接口

多线程第三种实现方式

  • 特点: 可以获取到多线程运行的结果
  • ①创建一个Callable接口的实现类
  • ②重写Callable接口中的call方法(返回多线程执行的结果)
  • ③创建Callable实现类对象(表示多线程要实现的任务)
  • ④创建Future实现类FutureTask的对象(用于管理多线程运行的结果)
  • ⑤创建Thread对象,并启动
/*** @author .29.* @create 2023-10-17 21:09*/
public class implementsCallable {//1.创建Callable接口实现类,泛型指定返回的线程执行结果的类型static class myCall implements Callable<Integer>{//2.重写实现类方法@Overridepublic Integer call() throws Exception {return 29;}}public static void main(String[] args) throws ExecutionException, InterruptedException {//3.实例化Callable实现类myCall call = new myCall();//4.创建FutureTask对象,管理多线程执行结果FutureTask<Integer> ft = new FutureTask<>(call);//5.创建线程,并启动Thread t1 = new Thread(ft);t1.start();//获取多线程运行的结果int result = ft.get();System.out.println(result);}
}

④ 比较优缺点

在这里插入图片描述




3. Thread类 常用方法

常用的成员方法

  • String getName():返回此线程名称

  • void setName(String name):修改此线程名称

  • static Thread currentThread():获取当前线程实例对象

  • static void sleep(long time):让线程休眠指定时间(单位:毫秒)

  • setPriority(int newPriority):设置线程的优先级

  • final int getPriority():获取此线程的优先级,线程默认的优先级为5,优先级范围1-10,数字越大,优先级越高。

  • final void setDaemon(boolean on):设置为守护线程,当其他的非守护线程执行完毕后,守护线程就没有存在的必要了,会陆续结束,不一定会执行完毕。

  • public static void yield():出让线程/礼让线程

    •     //1.自定义类,继承Thread类static class myThread extends Thread{//2.重写run方法,编写执行逻辑@Overridepublic void run() {for(int i = 0;i < 100; ++i){System.out.println("执行:" + getName());//出让CPU的执行权,即出让线程Thread.yield();}}}
      
  • public static void join()插入线程/插队线程,将此线程插入到当前线程之前,只有插入的线程执行完毕,当前线程才会执行。




4. 线程安全

1) 同步代码块

  • 格式:

    • synchronized(锁对象){//操作共享数据的代码
      }
      
    • 锁对象:可以是任意一个对象,但需要对象是唯一的,使用的锁不一样,没意义。建议使用当前类的字节码文件对象:Xxx.class

  • 特点:

    • ①锁默认是打开状态,有一个线程进去了,锁自动关闭。
    • ②里面的代码全部执行完毕,线程出来,锁自动打开。

2) 同步方法

  • 同步方法 —— 将synchronized关键字加到方法上。

  • 格式:

    • 修饰符 synchronized 返回值类型 方法名(方法参数){...}
      
  • 特点:

    • ①同步方法是锁住方法里面所有的代码。
    • ②锁对象不能自己指定
      • 非静态方法的锁对象:this(当前方法的调用者)
      • 静态方法的锁对象:当前类的字节码文件对象(Xxx.class)

3) Lock锁

  • 为了清晰表达如何加锁和释放锁,JDK5以后提供了一个新的锁对象Lock
  • Lock中提供了获取锁释放锁的方法:
    • void lock()——获得锁
    • void unlock()——释放锁
  • Lock是接口,获取Lock对象需要实例化Lock的实现类ReentrantLock
    • ReentrantLock()——构造方法、创建一个ReentrantLock实例

注意: 为了保证锁的释放,当调用lock()获得锁后,后面执行的代码放入到try{}catch{}块中,之后在finally{}块中调用unLock(),因为finally块中代码一定会执行,也就保证了锁一定会被释放。




5. 等待唤醒机制(生产者和消费者)

常用方法

  • void wait():当前线程等待,直至被其他线程唤醒
  • void notify():随机唤醒单个线程
  • void notifyAll():唤醒所有线程

1) 生产者

生产者 代码

/*** @author .29.* @create 2023-10-18 21:18*/
//继承Thread
public class Producer extends Thread{//重写run方法@Overridepublic void run() {//1.循环while(true){//2. 同步代码块synchronized(Product.lock){if(Product.count == 0){break;}else{if(Product.flag == 0){ //货架没有商品System.out.println("生产者生产商品"); //生产Product.flag = 1;                   //上架(修改状态)Product.lock.notifyAll();           //唤醒与锁绑定的所有线程}else{try {Product.lock.wait(); //货架有商品,等待} catch (InterruptedException e) {e.printStackTrace();}}}}}}
}

商品货架:

/*** @author .29.* @create 2023-10-18 21:20*/
public class Product {//总个数public static int count = 10;//消费者可否消费 0无产品,不可消费 1有产品,可消费public static int flag = 0;//锁对象public static Object lock = new Object();
}

2) 消费者

消费者 代码

/*** @author .29.* @create 2023-10-18 21:18*/
//继承Thread
public class Consumer extends Thread{//重写run方法@Overridepublic void run() {//1.循环while(true){//2.同步代码块synchronized(Product.lock){//3.判断货架是否上架了商品if(Product.flag == 0){//没有商品线程进入等待try {Product.lock.wait();} catch (InterruptedException e) {e.printStackTrace();}}else{//可以消费,消费者进行消费,总数-1Product.count--;System.out.println("消费者消费商品,还可消费"+Product.count+"件商品");//4. 通过锁,唤醒与锁绑定的所有线程Product.lock.notifyAll();//5. 修改货架状态Product.flag = 0;}}}}
}

测试:

/*** @author .29.* @create 2023-10-18 21:39*/
public class Test {public static void main(String[] args){Producer producer = new Producer();Consumer consumer = new Consumer();producer.start();consumer.start();}
}

运行

生产者生产商品
消费者消费商品,还可消费9件商品
生产者生产商品
消费者消费商品,还可消费8件商品
生产者生产商品
消费者消费商品,还可消费7件商品
生产者生产商品
消费者消费商品,还可消费6件商品
生产者生产商品
消费者消费商品,还可消费5件商品
生产者生产商品
消费者消费商品,还可消费4件商品
生产者生产商品
消费者消费商品,还可消费3件商品
生产者生产商品
消费者消费商品,还可消费2件商品
生产者生产商品
消费者消费商品,还可消费1件商品
生产者生产商品
消费者消费商品,还可消费0件商品

3) 等待唤醒 —— 阻塞队列方式

阻塞队列继承结构

  • Iterable 接口
  • Collection 接口
  • Queue 接口
  • BlockingQueue 接口
    • ArrayBlockingQueue 实现类
      • 底层是数组,有界
    • LinkedBlockingQueue实现类
      • 底层是链表,可看作无界,上限是int类型最大值

生产者 代码

/*** @author .29.* @create 2023-10-19 10:19*/
public class Producer extends Thread{private ArrayBlockingQueue<String> blockingQueue;public Producer(ArrayBlockingQueue<String> bq){blockingQueue = bq;}@Overridepublic void run() {while(true){//不断生产商品try {blockingQueue.put("商品");System.out.println("生产者生产了商品"); //输出语句不在线程同步范围内} catch (InterruptedException e) {e.printStackTrace();}}}
}

消费者 代码

/*** @author .29.* @create 2023-10-19 10:19*/
public class Consumer extends Thread{private ArrayBlockingQueue<String> blockingQueue;public Consumer(ArrayBlockingQueue<String> bq){blockingQueue = bq;}@Overridepublic void run() {while(true){//不断消费商品String take = null;try {//take(),获取阻塞队列元素,方法底层已经实现线程同步take = blockingQueue.take();System.out.println("消费者消费了 :" + take); //输出语句不在线程同步范围内} catch (InterruptedException e) {e.printStackTrace();}}}
}

测试

public class test {public static void main(String[] args){//实例化BlockingQueue对象,传入参数1,表示阻塞队列长度为1ArrayBlockingQueue<String> bq = new ArrayBlockingQueue<>(1);Consumer consumer = new Consumer(bq);Producer producer = new Producer(bq);consumer.start();producer.start();}
}

输出:

生产者生产了商品
生产者生产了商品
消费者消费了 :商品
消费者消费了 :商品
生产者生产了商品
生产者生产了商品
消费者消费了 :商品
消费者消费了 :商品
生产者生产了商品
生产者生产了商品
消费者消费了 :商品
消费者消费了 :商品
生产者生产了商品
生产者生产了商品
消费者消费了 :商品
消费者消费了 :商品
  • 输出存在重复现象,是因为输出语句不在线程同步范围内导致的,实际上put和take是线程安全的。



6. 线程状态

  • 1)新建(NEW),至今尚未启动的线程处于这种状态。
    • 创建线程对象后
  • 2)就绪(RUNNABLE),正在Java虚拟机中执行的线程处于这种状态。
    • 调用start()后
  • 3)阻塞(BLOCKING),受阻塞并等待某个监视器锁的线程处于这种状态。
    • 无法获得锁对象时
  • 4)等待(WAITING),无限期地等待另一个线程来执行某一特定操作的线程处于这种状态。
    • 调用wait()后
  • 5)计时等待(TIMED WAITING),等待另一个线程来执行取决于指定等待时间的操作的线程处于这种状态。
    • 调用sleep()后
  • 6)死亡(TERMINATED),已退出的线程处于这种状态。
    • 全部代码运行完毕后

在这里插入图片描述




7. 线程池

核心原理

  • ①创建一个空的线程池
  • ②提交任务时,线程池创建新的线程对象,执行完毕,线程归还线程池,等待下次任务
  • ③提交任务时若线程池没有空闲线程,也无法创建新的线程(线程池满了),任务就会排队等待。
  • 不断提交的任务,存在以下临界点:
      1. 当核心线程满时,再提交任务就会排队。
      2. 当核心线程满,队伍也满时,会创建临时线程。
      3. 核心线程满,队伍满,临时线程也满时,会触发任务拒绝策略。
    • 任务拒绝策略:
      • ThreadPoolExecutor.AbortPolicy:默认策略、丢弃任务并抛出RejectedExecutionException异常
      • ThreadPoolExecutor.DiscardPolicy:丢弃任务,但是不抛出异常,(不推荐)
      • ThreadPoolExecutor.DiscardOldestPolicy:抛弃队列中等待最久的任务,然后把当前任务加入等待队列。
      • ThreadPoolExecutor.CallerRunsPolicy:调用任务的run()方法绕过线程池直接执行。

实现线程池

  • Executors:线程池的工具类,通过调用方法返回不同类型的线程池对象。
    • public static ExecutorService newCachedThreadPool():创建一个没有上限的线程池。
    • public static ExcutorService newFixedThreadPool(int nThreads) 创建一个有上限的线程池,参数表示线程池的上限。
      • submit(Runnable/Callable<> in)方法:提交任务
      • shutdown()方法:销毁线程池

自定义线程池

  • 参数① 核心线程数量(不能小于0)
  • 参数② 最大线程数量(不能小于0,最大数量 >= 核心线程数)
  • 参数③ 空闲线程最大存活时间(不能小于0)
  • 参数④ 时间单位(用TimeUnit指定)
  • 参数⑤ 任务队列(不能为null)
  • 参数⑥ 创建线程工厂(不能为null)
  • 参数⑦ 任务的拒绝策略(不能为null)
  • 任务拒绝策略:
    • ThreadPoolExecutor.AbortPolicy:默认策略、丢弃任务并抛出RejectedExecutionException异常
    • ThreadPoolExecutor.DiscardPolicy:丢弃任务,但是不抛出异常,(不推荐)
    • ThreadPoolExecutor.DiscardOldestPolicy:抛弃队列中等待最久的任务,然后把当前任务加入等待队列。
    • ThreadPoolExecutor.CallerRunsPolicy:调用任务的run()方法绕过线程池直接执行。
/*** @author .29.* @create 2023-10-19 12:08*/
public class ThreadPoolDemo {public static void main(String[] args){ThreadPoolExecutor pool = new ThreadPoolExecutor(3,         //核心线程数6,                     //最大线程数,不能小于0,需要大于等于核心线程数60,                    //空闲线程最大存活时间TimeUnit.SECONDS,      //时间单位,用TimeUnit设置new ArrayBlockingQueue<>(3), //任务队列Executors.defaultThreadFactory(),     //创建线程工厂new ThreadPoolExecutor.AbortPolicy() //任务的拒绝策略(线程池内部类));pool.submit(new myCall());//提交任务pool.shutdown();//销毁线程池}
}



在这里插入图片描述

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

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

相关文章

人工智能基础_机器学习001_线性回归_多元线性回归_最优解_基本概念_有监督机器学习_jupyter notebook---人工智能工作笔记0040

线性和回归,就是自然规律,比如人类是身高趋于某个值的概率最大,回归就是通过数学方法找到事物的规律. 机器学习作用: 该专业实际应用于机器视觉、指纹识别、人脸识别、视网膜识别、虹膜识别、掌纹识别、专家系统、自动规划、智能搜索、定理证明、博弈、自动程序设计、智能控制…

【PyTorch】深度学习实践 01 Overview

人工智能概述 课程前置知识 线性代数 概率论 &#xff08;不要有路径依赖&#xff0c;遇到不会的就现学&#xff09;Python基础 人工智能 问题分类 人工智能&#xff0c;实际上就是利用计算机来代替人脑进行智能工作&#xff0c;计算机所要实现的智能可以分为两大部分&am…

python造测试数据存到excel

代码&#xff1a; from ExcelHandler import ExcelHandler from faker import Faker # 导入faker库的Faker方法 # ↓默认为en_US&#xff0c;只有使用了相关语言才能生成相对应的随机数据 fkFaker(locale"zh_CN")def create_date():m int(input(请输入要造的数据条…

PostgreSQL12中浮点数输出算法优化带来的小问题

最近碰到同事发来这样两个SQL&#xff0c;开发反馈输出的结果异常。 bill# select 0.1284*100::float;?column? --------------------12.839999999999998 (1 row)bill# select (0.1284*100)::float;float8 --------12.84 (1 row) 乍一看其实能看出明显的区别&#xff0c;由于…

通俗介绍:什么是 Redis ?

刚接触 Redis 的伙伴们可能会因为不熟悉而感到困惑。本文简述 Redis 是什么、有哪些作用的问题&#xff0c;是一篇短浅而入门级别的文章。 Redis官网&#xff1a;Redis 打开 Redis 官网可以看到&#xff0c;官方对 Redis 的介绍是这样的&#xff1a;The open source, in-memo…

pytorch C++ 移植

文章目录 前言安装 libtorch安装 opencv&#xff08;C&#xff09;模型转换通过跟踪转换为 Torch Script通过注解转换为 Torch Script 编写 C 代码编译环境搭建C 库管理方法一&#xff1a;手动配置 visual studio 环境方法二&#xff1a;cmake 配置环境 python 调用 C 程序 前言…

艾伏尼布:一种用于治疗复发或难治性AML和胆管癌的IDH1抑制剂【医游记】

&#xff08;图片来源于网络) 艾伏尼布&#xff0c;商品名为拓舒沃&#xff0c;是中国首个获批的针对异柠檬酸脱氢酶-1 (IDH1)突变癌症的强效口服靶向抑制剂1。这款药物最初由Agios Pharmaceuticals公司开发&#xff0c;2018年基石药业与Agios达成在大中华区开发及商业化的独家…

RK3399平台开发中安卓系统去除USB权限弹窗

RK3399平台开发中安卓系统去除USB权限弹窗 问题方法 问题 当我们在访问一个插入到Android系统的USB设备的时候往往是需要权限的&#xff0c;此时系统会弹出询问权限的对话框&#xff0c;而我们此时希望让它默认允许访问USB设备并且不希望用户看到这个对话框。 方法 文件目录&…

CSS - 常用属性和布局方式

目录 前言 一、常用属性 1.1、字体相关 1.2、文本相关 1.3、背景相关 1.3.1、背景颜色 1.3.2、背景图片 1.4、圆角边框 二、常用布局相关 2.1、display 2.2、盒子模型 2.2.1、基本概念 2.2.2、border 边框 2.2.3、padding 内边距 2.2.4、margin 外边距 2.3、弹…

gcc和makfile

gcc和makfile gcc预处理(进行宏替换)编译(生成汇编)汇编(生成机器可以识别的代码)连接(生成可以执行的文件或者库文件) makemakefile文件的编写 gcc 没安装的话&#xff0c;按照终端提示的安装命令安装就行 运行的格式&#xff1a; gcc [选项] [要编译的文件] [选项] [目标文件…

SylixOS BSP开发(八)

初始化FPU、MMU和Cache组件 本来想在不初始化这些部件的情况下把SylixOS先启动起来感受下&#xff0c;结果测试发现如果MMU不使能的话&#xff0c;系统启动过程中线程无法进行调度emm。。。所以只好把这一章节提前来讲了。这三个组件的初始化都是在bspInit.c中进行的。 1. FPU初…

Unity游戏开发中打造游戏攻击技能架构与设计

一、技能系统的设计 在 MOBA 游戏中&#xff0c;每个英雄角色都会有多个技能&#xff0c;这些技能可以分为普通攻击和技能攻击两种。普通攻击是英雄角色的基本攻击方式&#xff0c;而技能攻击则需要消耗一定的资源&#xff08;如蓝量&#xff09;才能使用。在设计技能系统时&a…

【面试经典150 | 链表】循环链表

文章目录 Tag题目来源题目解读解题思路方法一&#xff1a;哈希集合方法二&#xff1a;快慢指针方法三&#xff1a;计数 拓展其他语言python3 写在最后 Tag 【快慢指针】【哈希集合】【计数】【链表】 题目来源 141. 环形链表 题目解读 判断一个链表中是否存在环。 解题思路 …

(Java)中的数据类型和变量

文章目录 一、字面常量二、数据类型三、变量1.变量的概念2.语法的格式3.整型变量4.长整型变量5.短整型变量6.字节型变量 四、浮点型变量1.双精度浮点数2.单精度浮点数 五、字符型常量六、布尔型变量七、类型转换1.自动类型转换&#xff08;隐式&#xff09;2.强制类型转换(显式…

react的setState做了什么

1、为什么需要setState setState的作用是帮助我们更改数据的同时并且通知视图进行渲染。因为React并不会绑定视图和state&#xff0c;需要我们手动去更新视图。 2、setState什么时候是同步的&#xff0c;什么时候是异步的 setState这个方法在调用的时候是同步的&#xff0c;…

【人工智能Ⅰ】实验1:谓词表示法与产生式知识表示

实验1 谓词表示法与产生式知识表示 一、实验目的 1、熟悉谓词逻辑表示法&#xff1b; 2、理解和掌握产生式知识表示方法&#xff0c;实现产生式系统的规则库。 二、实验内容 要求通过C/C/python语言编程实现&#xff1a; 1、猴子摘香蕉问题 2、动物识别系统 &#xff08…

Spring无法加载静态属性和SpringBoot单元测试指定加载配置文件

一、Spring无法加载静态属性&#xff0c;怎么解决&#xff1f; Spring主要用于管理和注入Bean&#xff08;对象&#xff09;的实例属性&#xff0c;而不是静态属性。静态属性属于类本身&#xff0c;而不是类的实例&#xff0c;因此Spring的依赖注入机制不会处理它们。 看图&a…

项目部署Linux步骤

1、最小化安装centos7-环境准备 安装epel-release 安装epel-release&#xff0c;因为有些rpm包在官方库中找不到。前提是保证可以联网 yum install -y epel-release 修改IP net-tools net-tool&#xff1a;工具包集合&#xff0c;包含ifconfig等命令 yum install -y net-…

【MySQL-->数据操作】

文章目录 前言一、insert1.单行插入2.多行插入3.插入更新/替换 二、select1.全列查询2.指定列插入3.列别名4. 表达式计算5.去重6.where条件查询7.排序8.limit分页显示 三、update四、delete五、插入查询结果六、聚合函数六、聚合分组1.格式2.where和having的区别 前言 一、inse…

个人创业新机遇,零成本开启真人手办定制项目

桌上的日历变薄&#xff0c;2023年已经接近尾声&#xff0c;浅观这一年&#xff0c;您是否发现大家日常关注的重点有明显的变化&#xff0c;诸多社会事件和宏观数字的背后&#xff0c;潜藏着对经济的“不托底”&#xff0c;而当我们真正开始关注起用个人经济积累去对冲未来的不…