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

news2024/11/16 22:57:15

在这里插入图片描述

个人简介: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方法,编写执行逻辑
            @Override
            public 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.重写抽象方法
        @Override
        public 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.重写实现类方法
        @Override
        public 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方法,编写执行逻辑
              @Override
              public 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方法
    @Override
    public 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方法
    @Override
    public void run() {
        //1.循环
        while(true){
            //2.同步代码块
            synchronized(Product.lock){
                //3.判断货架是否上架了商品
                if(Product.flag == 0){
                    //没有商品线程进入等待
                    try {
                        Product.lock.wait();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }else{
                    //可以消费,消费者进行消费,总数-1
                    Product.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;
    }

    @Override
    public 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;
    }

    @Override
    public 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,表示阻塞队列长度为1
        ArrayBlockingQueue<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.coloradmin.cn/o/1133020.html

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

相关文章

Pytorch指定数据加载器使用子进程

torch.utils.data.DataLoader(train_dataset, batch_sizebatch_size, shuffleTrue,num_workers4, pin_memoryTrue) num_workers 参数是 DataLoader 类的一个参数&#xff0c;它指定了数据加载器使用的子进程数量。通过增加 num_workers 的数量&#xff0c;可以并行地读取和预处…

多伦多 Pwn2Own 大赛首日战报!三星 Galaxy S23 被黑两次

Bleeping Computer 网站披露&#xff0c;加拿大多伦多举行的 Pwn2Own 2023 黑客大赛的第一天&#xff0c;网络安全研究人员就成功两次攻破三星 Galaxy S23。 大会现场&#xff0c;研究人员还“演示"了针对小米 13 Pro 智能手机、打印机、智能扬声器、网络附加存储 (NAS) 设…

Ubuntu卸载或重置防火墙规则

Ubuntu卸载或重置防火墙规则 1、开启防火墙后查看对应规则编号&#xff0c;然后进行删除 sudo ufw status numbered ——查看所有规则编号id sudo ufw delete 2 ——删除对应id的规则&#xff08;比如删除2号规则&#xff09; 2、按规则来删除。 例如&#xff0c;如果你使用s…

Meetup 回顾|Data Infra 研究社第十六期(含资料发布)

本文整理于上周六&#xff08;10月21日&#xff09;Data Infra 第 16 期的活动内容。本次活动由 Databend 研发工程师-王旭东为大家带来了一场主题为《Databend hash join spill 设计与实现》的分享&#xff0c;让我们一起回顾一下吧~ 以下是本次活动的相关视频、资料及文字&a…

【算法】模拟退火算法(SAA,Simulated Annealing Algorithm)

模拟退火算法&#xff08;SAA&#xff09;简介 模拟退火算法&#xff08;SAA&#xff0c;Simulated Annealing Algorithm&#xff09;的灵感来源于工艺铸造流程中的退火处理&#xff0c;随着铸造温度升高&#xff0c;分子运动趋于无序&#xff0c;徐徐冷却后&#xff0c;分子运…

【数据分享】2014-2022年我国淘宝村点位数据(Excel格式/Shp格式)

电子商务是过去一二十年我国发展最快的行业&#xff0c;其中又以淘宝为代表&#xff0c;淘宝的发展壮大带动了一大批服务淘宝电子商务的村庄&#xff0c;这些村庄被称为淘宝村&#xff01; 截至到目前&#xff0c;阿里研究院梳理并公布了2014-2022年共9个年份的淘宝村名单&…

2.AUTOSAR SWC设计概述

1.SWC概述 SWC,全称Software Components,运行在RTE之上,属于应用算法逻辑这一层,如下图: 由1.AUTOSAR的架构及方法论中我们了解到该框架的提出就是为了减少平台移植成本、加快研发效率;这也就是说在AUTOSAR框架下,SWC作为组件是需要被重用的,意味着一个…

数据预处理(超详细)

import pandas as pd import numpy as np【例5-1】使用read_csv函数读取CSV文件。 df1 pd.read_csv("sunspots.csv")#读取CSV文件到DataFrame中 print(df1.sample(5))df2 pd.read_table("sunspots.csv",sep ",")#使用read_table&#xff0c;…

人工智能基础_机器学习003_有监督机器学习_sklearn中线性方程和正规方程的计算_使用sklearn解算八元一次方程---人工智能工作笔记0042

然后我们再来看看,如何使用sklearn,来进行正规方程的运算,当然这里 首先要安装sklearn,这里如何安装sklearn就不说了,自己查一下 首先我们还是来计算前面的八元一次方程的解,但是这次我们不用np.linalg.solve这个 解线性方程的方式,也不用 直接 解正规方程的方式: 也就是上面…

接口自动化测试实践

接口自动化概述 Python接口自动化测试零基础入门到精通&#xff08;2023最新版&#xff09; 众所周知&#xff0c;接口自动化测试有着如下特点&#xff1a; 低投入&#xff0c;高产出。 比较容易实现自动化。 和UI自动化测试相比更加稳定。 如何做好一个接口自动化测试项目呢…

华媒舍:怎样利用KOL出文营销推广打造出超级影响力?

利用KOL&#xff08;Key Opinion Leader&#xff09;出文营销推广已成为很多个人和企业提高影响力的重要方法。根据恰当的思路与技巧&#xff0c;你也能轻松吸引大批粉丝并打造无敌的存在影响力。下面我们就以科谱的形式详细介绍怎样利用KOL 出文营销推广&#xff0c;帮助自己做…

SD-WAN让跨境网络访问更快、更安全!

目前许多外贸企业都面临着跨境网络不稳定、不安全的问题&#xff0c;给业务合作带来了很多困扰。但是&#xff0c;现在有一个解决方案能够帮助您解决这些问题&#xff0c;让您的跨境网络访问更快、更安全&#xff0c;那就是SD-WAN&#xff01; 首先&#xff0c;让我们来看看SD-…

微机原理:逻辑运算指令、移位指令

文章目录 一、逻辑运算指令1、取反运算指令2、与运算指令3、或运算指令4、异或运算 二、移位指令1、开环移位指令算术左移&#xff1a;SHL、SAL算术右移&#xff1a;SAR逻辑右移&#xff1a;SHR 2、闭环移位指令含进位的循环左移&#xff1a;RCL含进位的循环右移&#xff1a;RC…

lunar-1.5.jar

公历农历转换包 https://mvnrepository.com/artifact/com.github.heqiao2010/lunar <!-- https://mvnrepository.com/artifact/com.github.heqiao2010/lunar --> <dependency> <groupId>com.github.heqiao2010</groupId> <artifactId>l…

使用文件附件

文件附件在peoplesoft中非常常见 This chapter provides an overview of the file attachment functions and discusses: Developing applications that use file attachment functions. Application development considerations. Application deployment and system configu…

基于 Appium 的 Android UI 自动化测试!

自动化测试是研发人员进行质量保障的重要一环&#xff0c;良好的自动化测试机制能够让开发者及早发现编码中的逻辑缺陷&#xff0c;将风险前置。日常研发中&#xff0c;由于快速迭代的原因&#xff0c;我们经常需要在各个业务线上进行主流程回归测试&#xff0c;目前这种测试大…

Kafka入门04——原理分析

目录 01理解Topic和Partition Topic(主题) Partition(分区) 02理解消息分发 消息发送到分区 消费者订阅和消费指定分区 总结 03再均衡(rebalance) 再均衡的触发 分区分配策略 RangeAssignor(范围分区) RoundRobinAssignor(轮询分区) StickyAssignor(粘性分区) Re…

软件测试面试1000问(含文档)

前前后后面试了有20多家的公司吧&#xff0c;最近抽空把当时的录音整理了下&#xff0c;然后给大家分享下 开头都是差不多&#xff0c;就让做一个自我介绍&#xff0c;这个不用再给大家普及了吧 同时&#xff0c;我也准备了一份软件测试视频教程&#xff08;含接口、自动化、…

进阶课4——随机森林

1.定义 随机森林是一种集成学习方法&#xff0c;它利用多棵树对样本进行训练并预测。 随机森林指的是利用多棵树对样本进行训练并预测的一种分类器&#xff0c;每棵树都由随机选择的一部分特征进行训练和构建。通过多棵树的集成&#xff0c;可以增加模型的多样性和泛化能力。…

MTK AEE_EXP调试方法及user版本打开方案

一、AEE介绍 AEE (Android Exception Engine)是安卓的一个异常捕获和调试信息生成机制。 手机发生错误(异常重启/卡死)时生成db文件(一种被加密过的二进制文件)用来保存和记录异常发生时候的全部内存信息,经过调试和仿真这些信息,能够追踪到异常的缘由。 二、调试方法…