【Java 第九篇章】多线程实际工作中的头大的模块

多线程是一种编程概念,它允许多个执行路径(线程)在同一进程内并发运行。

一、多线程的概念和作用

1、概念

  • 线程是程序执行的最小单元,一个进程可以包含多个线程。每个线程都有自己的程序计数器、栈和局部变量,但它们共享进程的内存空间和其他资源(如打开的文件、网络连接等)。
  • 多线程就是在一个程序中同时运行多个线程,每个线程可以执行不同的任务或相同任务的不同部分。

2、作用

  • 提高程序性能:通过将一个大任务分解成多个小任务并在不同线程中并行执行,可以充分利用多核处理器的优势,减少程序的执行时间。例如,在图像编辑软件中,可以使用一个线程来处理用户界面的交互,另一个线程用于后台的图像渲染,从而提高整体的响应速度。

  • 增强程序的响应能力:在一些需要与用户进行实时交互的应用中,多线程可以确保即使在执行耗时操作时,程序仍然能够及时响应用户的输入。比如在网页浏览器中,一个线程可以用于加载网页内容,而另一个线程可以处理用户的鼠标点击和滚动操作,这样用户不会感觉到界面卡顿。

  • 实现异步操作:多线程使得程序可以在后台执行某些操作,而不阻塞主线程的执行。例如,在文件下载程序中,启动一个新线程来下载文件,主线程可以继续显示下载进度或执行其他任务,当下载完成后,再进行相应的处理。

3、线程和进程

  • 线程:一个进程中可以有多个线程,如看视频的同时可以听声音,看图像,看弹幕等等。
  • 进程: 操作系统中运行的程序就是进程,比如QQ、播放器、游戏、IDE等等。

二、实现多线程的方式

1、继承 Thread 类

定义一个类继承自Thread类,然后重写run方法,在run方法中编写线程要执行的任务代码。

public class MyThread extends Thread {@Overridepublic void run() {System.out.println("This is a new thread.");}
}

启动线程

public class Main {public static void main(String[] args) {MyThread myThread = new MyThread();myThread.start();}
}

当调用start方法时,Java 虚拟机会自动调用该线程的run方法来执行线程的任务。

2、实现Runnable接口

定义一个类实现Runnable接口,并实现run方法:

public class MyRunnable implements Runnable {@Overridepublic void run() {System.out.println("This is a thread implemented by Runnable.");}
}

启动线程

public class Main {public static void main(String[] args) {MyRunnable myRunnable = new MyRunnable();Thread thread = new Thread(myRunnable);thread.start();}
}

这种方式更加灵活,因为 Java 不支持多继承,如果一个类已经继承了其他类,就不能再继承Thread类了,此时可以采用实现Runnable接口的方式来创建线程。

3、实现 Callable 接口

在 Java 中,Callable接口是一种用于创建可以返回结果并且可能抛出异常的任务的方式。它与Runnable接口类似,但Callable可以返回结果,而Runnable的run方法没有返回值。

定义一个实现Callable接口的类

import java.util.concurrent.Callable;public class MyCallable implements Callable<String> {@Overridepublic String call() throws Exception {// 这里编写具体的任务代码,此示例中模拟一个耗时操作后返回结果Thread.sleep(2000);return "Callable 任务执行完成并返回结果";}
}

在主线程中使用Callable

import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;public class Main {public static void main(String[] args) {// 创建一个线程池ExecutorService executorService = Executors.newSingleThreadExecutor();// 提交 Callable 任务到线程池并获取 Future 对象Future<String> future = executorService.submit(new MyCallable());try {// 从 Future 对象获取 Callable 任务的结果,如果任务未完成,此方法会阻塞直到任务完成String result = future.get();System.out.println(result);} catch (InterruptedException | ExecutionException e) {e.printStackTrace();}// 关闭线程池executorService.shutdown();}
}

在上述代码中:

  • 首先定义了MyCallable类实现Callable接口,并重写call方法来执行具体的任务,这里模拟了一个耗时操作(通过Thread.sleep)后返回一个字符串结果。
  • 在main方法中,创建了一个单线程的线程池ExecutorService,通过submit方法将MyCallable任务提交到线程池中执行,submit方法会返回一个Future对象,这个对象可以用来获取任务的执行结果。然后通过future.get方法获取任务的结果,如果任务还未完成,get方法会阻塞当前线程直到任务完成。最后关闭线程池。

使用Callable接口可以方便地在多线程环境中执行有返回值的任务,并且可以通过Future对象来管理任务的执行状态和结果。这在需要执行一些耗时的计算并获取结果的场景中非常有用,比如在网络请求、数据库查询等操作中。

三、线程的五种状态

线程存在五种状态分别是:创建状态、就绪状态、阻塞状态、死亡状态、运行状态。

在这里插入图片描述

四、多线程的优点和挑战

1、优点

  • 资源共享:由于线程共享进程的内存空间,它们可以方便地共享数据和资源。这使得在多个线程之间传递信息和协作变得相对容易。例如,多个线程可以同时访问和修改同一个数组,而不需要进行复杂的数据传递和复制操作。
  • 提高效率:如前面提到的,多线程能够充分利用多核处理器的优势,将任务并行执行,从而提高程序的整体效率。特别是对于计算密集型和 I/O 密集型任务,多线程可以显著减少执行时间。
  • 简化程序结构:对于一些复杂的应用程序,将任务分解为多个线程可以使程序的结构更加清晰和易于维护。每个线程可以专注于执行一个特定的子任务,使得代码的逻辑更加模块化。

2、挑战

  • 线程安全问题:当多个线程同时访问和修改共享数据时,可能会导致数据不一致或程序错误。例如,两个线程同时对一个计数器进行递增操作,如果不采取适当的同步措施,可能会导致计数器的值不准确。常见的解决方法包括使用synchronized关键字、Lock接口等进行线程同步。
  • 死锁:在多线程编程中,如果多个线程相互等待对方持有的资源,就会导致死锁。例如,线程 A 持有资源 X 并等待资源 Y,而线程 B 持有资源 Y 并等待资源 X,这时两个线程都无法继续执行,程序就会陷入死锁状态。避免死锁需要合理的资源分配策略和线程同步设计。
  • 线程调度复杂性:操作系统负责线程的调度,决定哪个线程何时可以执行。然而,线程的调度是不可预测的,这可能导致程序的执行顺序不确定。例如,在一个多线程的游戏程序中,如果线程调度导致游戏逻辑的执行顺序不一致,可能会出现画面闪烁或游戏状态异常等问题。这就需要程序员在编写多线程程序时,充分考虑到各种可能的执行顺序,并进行适当的处理。

TODO: 未解决的问题

  • 线程的方法:
    • 停止线程
    • 休眠线程
    • 礼让线程
    • 合并线程
  • 线程优先级:
  • 守护进程:
  • 线程同步:
  • 同步方法:
    • synchronized
  • 死锁:
  • Lock锁:
  • 生产者和消费者:
    • 解决方式:
      • 利用缓冲去解决:管道法
      • 信号灯法:标志位
  • 线程池:

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

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

相关文章

Python获取Excel内容

Python获取Excel内容 目录 Python获取Excel内容1.读取Excel并登陆2.下载Excel中图片 数据存储到列表3.上传到接口 需求&#xff1a;获取xlsx files目录下的所有Excel信息&#xff0c;并将数据打包成字典格式上传到接口 示例数据&#xff1a; 1.读取Excel并登陆 import os impo…

【算法】贪心算法

应用场景——集合覆盖问题 假设存在下面需要付费的广播台&#xff0c;以及广播台信号可以覆盖的地区。如何选择最少的广播台&#xff0c;让所有的地区都可以接收到信号 贪心算法介绍 1.贪心算法是指在对问题进行求解时&#xff0c;在每一步选择中都采取最好或者最优的选择 2…

智观察 | 行业赛道里的AI大模型

‍ “AI改变世界”被炒得热火朝天&#xff0c;结果就换来AI聊天&#xff1f; 实际上&#xff0c;在日常娱乐之下&#xff0c;AI正在暗暗“憋大招”&#xff0c;深入各行各业&#xff0c;发挥更专业的作用。 自动驾驶 最近“萝卜快跑”霸榜热搜长达一周&#xff0c;让无人驾…

手机在网状态接口如何对接?(二)

一、什么是手机在网状态&#xff1f; 传入手机号码&#xff0c;查询该手机号的在网状态&#xff0c;返回内容有正常使用、停机、在网但不可用、不在网&#xff08;销号/未启用/异常&#xff09;、预销户等多种状态。 二、手机在网状态使用场景&#xff1f; 1.用户验证与联系…

【问题解决方案】npm install报错问题:npm ERR! - 多种解决方案,总有一种可以解决

文章目录 1.问题重述2.解决方案方案1.确认根目录正确方案2.确认文件名正确方案3. 确认node.js安装完成&#xff08;注意这个环境变量配置没有写完&#xff09;方案4 改用yarn安装&#xff08;亲测可用&#xff09; 3.延申问题解决方案问题1&#xff1a;需要低版本的node.js 写在…

企业图纸防泄密怎么做?最好的八款图纸加密软件推荐

保护企业图纸不被泄露是现代企业信息安全管理中的重要任务。随着信息技术的发展&#xff0c;企业需要采取多种措施来确保图纸的安全性。以下是一些常用的图纸防泄密方法和八款推荐的图纸加密软件&#xff1a; 图纸防泄密方法 1. 数据备份&#xff1a;定期备份图纸数据&#xf…

Jboss 漏洞

一.CVE-2015-7501 访问/invoker/JMXInvokerServlet 开启下载存在漏洞 二.CVE-2017-7504 三CVE-2017-12149 启动vulhub环境&#xff0c;访问/invoker/readonly出现如下界面&#xff0c;说明存在漏洞 使用工具连接 四.Administration Console弱⼝令 访问/admin-console/login…

数据库的管理

1、官网下载或者wget tar -xvf mysql-8.0.33-1.el7.x86_64.rpm-bundle.tar 2、确定mysql-community-server正常安装之后就可以开始配置 3、初始化mysqld 服务 mysqld initeialize 4、启动服务 systemctl start mysqld 5、添加开机启动列表 systecmctrl enable mysqld在/var…

git——Git提交本地项目代码到远程Github仓库步骤图解

目录 一、Git提交本地项目代码到远程Github仓库步骤 一、Git提交本地项目代码到远程Github仓库步骤 1、在Github创建一个空仓库&#xff0c;例如名称为jetcache-demo 2、打开【Git Bash Here】 3、进入本地项目文件夹 cd d:/ cd D:/project1/本地服务/jetcache-demo4、初始…

Golang面试题三(map)

1.map底层实现 由图看出&#xff0c;其实map的底层结构体是hmap&#xff0c;同时hmap里面维护着若干个bucket数组&#xff08;即桶数组&#xff09;。bucket数组中每个元素都是bmap结构的&#xff0c;bmap中存储着8个key-value的键值对&#xff0c;如果是满了的话&#xff0c;当…

用OpenCV与MFC写一个简单易用的图像处理程序

工厂里做SOP及测试报告以及员工资格鉴定等常需用到简单的图像处理&#xff0c;PS等软件正版费用不菲&#xff0c;学习起来成本也高。Windows自带的图像处理软件&#xff0c;用起来也不是那么得心应手。因此我用OpenCV与MFC写了一个简单易用的图像处理程序。 程序界面 基于简单…

从传统监控到智能化升级:EasyCVR视频汇聚平台的一站式解决方案

随着科技的飞速发展和社会的不断进步&#xff0c;视频监控已经成为现代社会治安防控、企业管理等场景安全管理中不可或缺的一部分。而在视频监控领域&#xff0c;EasyCVR视频汇聚平台凭借其强大的多协议接入能力&#xff0c;在复杂多变的网络环境中展现出了卓越的性能和广泛的应…

【第15章】Spring Cloud之Gateway网关过滤器(URL黑名单)

文章目录 前言一、常用网关过滤器1. 常用过滤器2. 示例3. Default Filters 二、定义接口服务1. 定义接口 三、自定义过滤器1. 过滤器类2. 应用配置 四、单元测试1. 正常2. 黑名单 总结 前言 上一章我们通过&#xff0c;路由断言根据请求IP地址的黑名单功能&#xff0c;作用范围…

【C#语音文字互转】C#语音转文字(方法一)

Whisper.NET开源项目&#xff1a;https://github.com/sandrohanea/whisper.net/tree/main 一. 环境准备 在VS中安装 Whisper.net&#xff0c;在NuGet包管理器控制台中运行以下命令&#xff1a; Install-Package Whisper.net Install-Package Whisper.net.Runtime其中运行时包…

STL-queue容器适配器

目录 一、queue 1.1 使用 1.2 模拟实现 二、priority_queue 2.1 使用 2.2 仿函数 2.2.1 概念 2.2.2 使用 2.3 模拟实现 一、queue 1.1 使用 具体解释详见官方文档&#xff1a;queue - C Reference (cplusplus.com) queue就是数据结构中的队列&#xff1a;数据结构之…

深度学习中降维的几种方法

笔者在搞网络的时候碰到个问题&#xff0c;就是将特征维度从1024降维到268&#xff0c;那么可以通过哪些深度学习方法来实现呢&#xff1f; 文章目录 1. 卷积层降维2. 全连接层降维3. 使用注意力机制4. 使用自编码器 1. 卷积层降维 可以使用1x1卷积层&#xff08;也叫pointwis…

《大道平渊》· 拾柒 —— 个人的心理定位决定市场

《大道平渊》 拾柒 个人的心理定位决定市场。 对于个人定位来说&#xff0c;个人的心理定位影响你的行为。 比如我的心理定位是经营者&#xff0c;那我的行为则是满足市场需求和解决问题。 因为心理定位的不同&#xff0c;会影响你思考问题的角度。 . 以上皆为个人思考&am…

【为什么不要买运营商的机顶盒?解锁智能电视新体验,从一台刷机机顶盒开始】

【置顶:机顶盒刷机步骤请跳转此链接】 在这个数字化飞速发展的时代&#xff0c;电视早已不再是单一的播放工具&#xff0c;它正逐步演变成为家庭娱乐与信息获取的综合中心。然而&#xff0c;许多家庭在选择机顶盒时&#xff0c;往往会因为惯性或便利而直接选择运营商提供的机顶…

常见中间件漏洞(三、Jboss合集)

目录 三、Jboss Jboss介绍 3.1 CVE-2015-7501 漏洞介绍 影响范围 环境搭建 漏洞复现 3.2 CVE-2017-7504 漏洞介绍 影响范围 环境搭建 漏洞复现 3.3 CVE-2017-12149 漏洞简述 漏洞范围 漏洞复现 3.4 Administration Console弱囗令 漏洞描述 影响版本 环境搭建…

【多线程-从零开始-伍】volatile关键字和内存可见性问题

volatile 关键字 import java.util.Scanner; public class Demo2 { private static int n 0; public static void main(String[] args) { Thread t1 new Thread(() -> { while(n 0){ //啥都不写 } System.out.println("t1 线程结束循环"); }, "…