Java之多线程和并发应用

news2024/11/18 7:23:55

多线程和并发应用

  • 1. 概念与意义
  • 2. 多线程的实际应用场景
    • 2.1 网络通信
    • 2.2 数据库操作
    • 2.3 图片处理
  • 3. 多线程的创建方式
    • 3.1 继承 Thread 类
    • 3.2 实现 Runnable 类
    • 3.3 实现 Callable 接口
    • 3.4 比较 Thread 和 Runnable
    • 3.5 线程同步与锁
    • 3.6 线程池
  • 4. 并发编程的挑战和解决方案
    • 4.1 线程安全性
    • 4.2 死锁
  • 5. 实际案例:并发下载器
  • 6. 总结

1. 概念与意义


在这里插入图片描述

  • 多线程是指一个进程内创建多个线程,使得程序能够同时执行多个任务,从而充分利用多核处理器和资源。
  • 并发是指多个任务交替执行,从宏观上看,任务似乎是同时进行的。

这两者结合,可以提高程序的效率和性能,实现更好的用户体验。

2. 多线程的实际应用场景

2.1 网络通信


在网络通信中,多线程可以实现同时接受多个客户端请求。例如,一个聊天服务器可以利用多线程,让每个连接的客户端都能在独立的线程中进行通信。这样一方面提高了服务器的响应速度,另一方面不会因为一个客户端的阻塞而影响其他客户端。

2.2 数据库操作


数据库操作通常是I/O密集型任务。使用多线程可以在进行数据库查询的同时,进行其他任务,提高系统的吞吐量。例如,在一个电子商务系统中,可以使用多线程同时查询不同商品的库存和价格,然后将结果整合给用户。

2.3 图片处理


图片处理是计算密集型任务。例如,一款图片编辑软件可以在后台使用多线程进行图像的滤镜处理、缩放等操作,让用户可以同时进行多个编辑操作,而不会感觉卡顿。

3. 多线程的创建方式

3.1 继承 Thread 类


  1. 自定义线程类继承 Thread
  2. 重写 run() 方法,编写线程执行体
  3. 创建线程对象,调用 start() 方法启动线程
public class MyThread extends Thread {
    @Override
    public void run() {
        // run方法线程体
        for (int i = 0; i < 20; i++) {
            System.out.println("线程执行中..." + i);
        }
    }

    public static void main(String[] args) {
        // 创建一个线程对象
        MyThread myThread = new MyThread();
        // 调用start()开启线程
        myThread.start();
        // main线程
        for (int i = 0; i < 200; i++) {
            System.out.println("main线程执行中..." + i);
        }
    }
}

注意: 线程不一定立即执行,CPU安排调度

3.2 实现 Runnable 类


public class RunnableThread implements Runnable {
    @Override
    public void run() {
        for (int i = 0; i < 20; i++) {
            System.out.println("线程体执行中" + i);
        }
    }

    public static void main(String[] args) {
        RunnableThread runnableThread = new RunnableThread();
        // 创建线程对象并启动线程
        new Thread(runnableThread).start();
        // 执行main线程
        for (int i = 0; i < 200; i++) {
            System.out.println("main线程执行中" + i);
        }
    }
}

3.3 实现 Callable 接口


public class CallableThread implements Callable {
    @Override
    public String call() throws Exception {
        for (int i = 0; i < 20; i++) {
            System.out.println("线程体正在执行中" + i);
        }
        return "执行成功";
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        CallableThread callableThread = new CallableThread();
        // 创建执行服务,江县城放进去
        ExecutorService executorService = Executors.newFixedThreadPool(3);
        // 执行提交
        Future<String> submit = executorService.submit(callableThread);
        // 获取结果
        Object o = submit.get();
        System.out.println(o.toString());
        // 关闭结果
        executorService.shutdown();
    }
}

3.4 比较 Thread 和 Runnable


后者更推荐,因为它可以避免 Java 单继承的限制,且提供更好的代码复用性。

3.5 线程同步与锁


在多线程环境下,可能会出现数据竞争和并发问题。Java 提供了synchronized 关键字和ReentrantLock 等机制来实现线程同步和锁定,确保多个线程对共享资源的安全访问。

3.6 线程池


Java 的线程池可以管理线程的创建、复用和销毁,避免频繁地创建和销毁线程。这在高并发环境下非常有用,提高了性能和资源利用率。

ExecutorService executor = Executors.newFixedThreadPool(5);
for (int i = 0; i < 10; i++) {
    executor.execute(new MyRunnable());
}
executor.shutdown();

线程池的执行流程
在这里插入图片描述

使用线程池的好处

  1. 降低资源消耗。通过重复利用已创建的线程,降低线程创建和销毁造成的消耗
  2. 提高响应速度。当任务到达时,任务可以不需要等到线程创建就能立即执行。
  3. 增加线程的可管理性。线程是稀缺资源,使用线程池可以进行统一分配,调优和监控。

4. 并发编程的挑战和解决方案

4.1 线程安全性


多线程并发访问共享资源时,可能会导致数据不一致性。通过使用锁、原子操作等手段,可以确保多线程环境下的数据安全。

4.2 死锁


死锁是指多个线程相互等待对方释放资源,从而导致程序无法继续进行。避免死锁可以通过破坏死锁的四个必要条件之一来实现,如加锁顺序一致性、资源分配有序性等。

5. 实际案例:并发下载器


考虑一个并发下载器,它可以同时下载多个文件,提高下载效率。每个文件下载可以在独立的线程中进行,下载完成后汇总结果。这种应用可以通过线程池和任务划分来实现。

  1. 导入依赖
<!-- 导入多线程案例下载图片 io包 -->
<dependency>
    <groupId>commons-io</groupId>
    <artifactId>commons-io</artifactId>
    <version>2.8.0</version>
</dependency>
  1. 创建线程,执行任务
public class ExampleCallable implements Runnable {
    private String url;
    private String name;

    public ExampleCallable(String url, String name) {
        this.url = url;
        this.name = name;
    }


    @Override
    public void run() {
        WebDownloader webDownloader = new WebDownloader();
        webDownloader.download(url, name);
        System.out.println("下载文件名为:" + name);
    }

    // 下载器
    class WebDownloader {
        // 下载方法
        public void download(String url, String name) {
            try {
                FileUtils.copyURLToFile(new URL(url), new File((name)));
            } catch (IOException e) {
                System.out.println("io 执行出现异常");
                throw new RuntimeException(e);
            }
        }
    }

    public static void main(String[] args) throws ExecutionException, InterruptedException {
        ExampleCallable ec1 = new ExampleCallable("https://img-home.csdnimg.cn/images/20201124032511.png", "1.png");
        ExampleCallable ec2 = new ExampleCallable("https://img-home.csdnimg.cn/images/20201124032511.png", "2.png");
        ExampleCallable ec3 = new ExampleCallable("https://img-home.csdnimg.cn/images/20201124032511.png", "3.png");

        // 创建线程池
        ExecutorService executorService = Executors.newFixedThreadPool(3);
        // 提交任务
        executorService.submit(ec1);
        executorService.submit(ec2);
        executorService.submit(ec3);
        // 关闭线程池
        executorService.shutdown();
    }
}

6. 总结


多线程编程和并发技术在Java中具有广泛的应用,从网络通信到图像处理,从数据库操作到高并发服务。通过合理的线程管理和同步机制,可以实现更高效的程序和更好的用户体验。然而,多线程编程也带来了一些挑战,如线程安全性和死锁。掌握多线程编程和基本概念和实际应用,将有助于你构建更强大、更可靠的应用程序。

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

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

相关文章

mysql存储过程定时调度

假设我们要创建一个简单的数据库&#xff0c;其中包含两张表&#xff1a;students 表和 courses 表&#xff0c;以及一个存储过程用于插入学生数据。下面是完整的建表语句、插入语句和存储过程&#xff1a; 1】建表 -- 创建 courses 表 CREATE TABLE courses (course_id INT …

【自然语言处理】大模型高效微调:PEFT 使用案例

文章目录 一、PEFT介绍二、PEFT 使用2.1 PeftConfig2.2 PeftModel2.3 保存和加载模型 三、PEFT支持任务3.1 Models support matrix3.1.1 Causal Language Modeling3.1.2 Conditional Generation3.1.3 Sequence Classification3.1.4 Token Classification3.1.5 Text-to-Image Ge…

OpenLayers入门,OpenLayers涂鸦手绘线条、圆形和多边形,涂鸦线条自动收尾连接成多边形

专栏目录: OpenLayers入门教程汇总目录 前言 本章再次讲解OpenLayers绘制图形功能,上一章中《OpenLayers图形绘制,OpenLayers实现在地图上绘制线段、圆形和多边形》我们已经讲过多种图形的绘制,本章主要讲解自由涂鸦手绘绘制线条,圆形和任意形状。 二、依赖和使用 &q…

Spring BeanPostProcessor 接口的作用和使用

BeanPostProcessor 接口是 Spring 框架中的一个扩展接口&#xff0c;用于在 Spring 容器实例化、配置和初始化 bean 的过程中提供自定义的扩展点。通过实现这个接口&#xff0c;您可以在 bean 实例创建的不同生命周期阶段插入自己的逻辑&#xff0c;从而实现对 bean 行为的定制…

微仓助力制造业数字化转型,实现高效智能供应链!

随着制造业数字化转型的浪潮迅猛涌现&#xff0c;为了应对日益复杂和快速变化的市场需求&#xff0c;制造企业面临着提高生产效率、降低成本、实现供应链高效运作的挑战。在这个数字化时代&#xff0c;微仓作为智能仓储解决方案&#xff0c;正发挥着关键的作用&#xff0c;成为…

verity cannot ... ‘/dev/block/dm-4‘ is read-only/ 证书cacerts系统目录

网上的说的一种做法是 su mount -o rw,remount / mount -o rw,remount /system cp /data/misc/user/0/cacerts-added/269953fb.0 /system/etc/security/cacerts/ rm /data/misc/user/0/cacerts-added/269953fb.0 reboot 但是我机子是android 12.0 提示 /dev/block/dm-4…

NKD:容器云集群与 OS 一体化运维利器

NKD 是 NestOS-kubernetes-Deployer 的缩写&#xff0c;是为了基于 NestOS 部署的 Kubernetes 集群运维工作准备的解决方案。其目标是在集群外提供对集群基础设施&#xff08;包括操作系统和 Kubernetes 基础组件&#xff09;的部署、更新和配置管理等服务。 1. 引言 Kuberne…

双栏排版中,[!b]等等的一些命令都不行啊

\usepackage{stfloats} Latex&#xff1a;图片、表格占据双栏排版的两栏时 的位置控制_latex stfloats_weixin_39450145的博客-CSDN博客目录1.问题&#xff1a;怎么在双栏排版中&#xff0c;让占据两栏的表格出现在页面顶端&#xff1f;2.解决&#xff1a;1&#xff09;图片&a…

378. 有序矩阵中第 K 小的元素

378. 有序矩阵中第 K 小的元素 原题链接&#xff1a;完成情况&#xff1a;解题思路&#xff1a;参考代码&#xff1a;__378有序矩阵中第K小的元素__直接排序__378有序矩阵中第K小的元素__归并排序__378有序矩阵中第K小的元素__二分查找 原题链接&#xff1a; 378. 有序矩阵中…

中学高级本习集c++

第一章 回溯法 1.1 马拦过河卒 源程序名 knight.???&#xff08;pas, c, cpp&#xff09; 可执行文件名 knight.exe 输入文件名 knight.in 输出文件名 knight.out 【问题描述】 棋盘上A点有一个过河卒&#xff0c;需要走到目标B点…

山西电力市场日前价格预测【2023-08-10】

日前价格预测 预测明日&#xff08;2023-08-10&#xff09;山西电力市场全天平均日前电价为328.01元/MWh。其中&#xff0c;最高日前电价为366.62元/MWh&#xff0c;预计出现在20: 00。最低日前电价为283.28元/MWh&#xff0c;预计出现在13: 15。 价差方向预测 1&#xff1a; 实…

本质安全设备标准(IEC60079-11)的理解(六)温度

本质安全设备标准&#xff08;IEC60079-11&#xff09;的理解&#xff08;六&#xff09;温度 本质安全设备的温度要求也是非常复杂的&#xff0c;首先在标准中涉及有3个温度的概念&#xff1a; 环境温度ambient temperature&#xff0c; 工作温度service temperature和最高表…

聊聊机器人学习中的“资产”与“负债”(节选)

前言&#xff1a; 程序里面的bug&#xff0c;人生过程的bug&#xff0c;都非常让人烦躁的。 前一篇&#xff1a; 聊聊机器人学习中的“投资”与“消费”&#xff08;节选&#xff09; 有很多学生和朋友&#xff0c;越学越累&#xff0c;越活越累&#xff1f;这到底是为何&a…

Java | 包和内部类

目录 一、Java类包 1.1 类名冲突 1.2 完整的类路径 1.3 创建包 1.4 导入包 1.4.1 使用import关键字导入包 1.4.2 使用import导入静态成员 二、内部类 2.1 成员内部类 2.1.1 成员内部类简介 2.1.2 使用this关键字获取内部类与外部类的引用 2.2 匿名内部类 类除了具有…

无涯教程-Perl - getnetbyname函数

描述 此函数返回由NAME指定的网络信息(在列表context中)($name,$aliases,$addrtype,$net) 语法 以下是此函数的简单语法- getnetbyname NAME返回值 此函数在错误时返回undef,否则在标量context中返回网络地址,在错误时返回空列表,否则在列表context中返回网络记录(名称,别…

在软件测试中,如何有效地跟踪和管理缺陷?

在软件测试中&#xff0c;跟踪和管理缺陷是非常重要的&#xff0c;因为这有助于确保所有问题得到妥善处理&#xff0c;避免在产品发布后出现问题。以下是跟踪和管理缺陷的一些有效方法&#xff1a; 1.创建缺陷报告&#xff1a;当发现一个缺陷时&#xff0c;应该立即创建一个缺…

前端进阶js02----null和undefined的区别

1.相同点 1)都是原始类型的值且保存在栈中。 2) 在布尔运算中都会被认为是false 2.不同点 1&#xff09;null是js的关键字&#xff0c;表示空值&#xff1b;undefined不是关键字&#xff0c;是一个全局变量。 2&#xff09;值相同&#xff0c;但类型不一样 值相同&#xff1a…

Day 25 C++ queue 容器(队列)

文章目录 queue 基本概念定义注意基本概念队头&#xff08;Front&#xff09;——指向队列中最早添加的元素的位置。队尾&#xff08;Rear&#xff09;——指向队列中最后添加的元素的位置。入队&#xff08;Enqueue&#xff09;——将元素添加到队尾。出队&#xff08;Dequeue…

Docker 方式 部署 vue 项目 (docker + vue + nginx)

1.安装好 nginx 。 2. 把 vue 项目的源码克隆到确定目录下。用 git 管理&#xff0c;所以直接 git clone 到既定目录就行了。 如我的目录是&#xff1a;/root/jiangyu/projects/gentle_vue/gentle_vue_code 。 3. 项目打包&#xff1a; npm run build 复制 会自动生成 dist…

【Linux】-进程概念之进程优先级(如何去进行调度以及进程切换),还不进来看看??

&#x1f496;作者&#xff1a;小树苗渴望变成参天大树&#x1f388; &#x1f389;作者宣言&#xff1a;认真写好每一篇博客&#x1f4a4; &#x1f38a;作者gitee:gitee✨ &#x1f49e;作者专栏&#xff1a;C语言,数据结构初阶,Linux,C 动态规划算法&#x1f384; 如 果 你 …