《Java 简易速速上手小册》第6章:Java 并发编程(2024 最新版)
文章目录
- 6.1 线程的创建和管理 - 召唤你的士兵
- 6.1.1 基础知识
- 6.1.2 重点案例:实现一个简单的计数器
- 6.1.3 拓展案例 1:定时器线程
- 6.1.4 拓展案例 2:使用 Executor 框架管理线程
- 6.2 同步机制 - 维持军队的秩序
- 6.2.1 基础知识
- 6.2.2 重点案例:银行转账操作
- 6.2.3 拓展案例 1:生产者消费者问题
- 6.2.4 拓展案例 2:读写锁实现缓存系统
- 6.3 并发工具类 - 你的特殊武器
- 6.3.1 基础知识
- 6.3.2 重点案例:使用 CountDownLatch 协调任务
- 6.3.3 拓展案例 1:使用 CyclicBarrier 同步周期性任务
- 6.3.4 拓展案例 2:使用 Semaphore 控制资源访问
6.1 线程的创建和管理 - 召唤你的士兵
在Java并发编程的世界中,线程是执行任务的基本单位。正确地创建和管理线程就像是召唤和指挥你的士兵一样重要。让我们深入探索如何有效地召唤这些勇士,并确保他们能够有效地完成任务。
6.1.1 基础知识
-
创建线程的两种方式:
- 继承Thread类:创建一个新类继承
Thread
类,并重写run()
方法。通过实例化这个类并调用其start()
方法来启动线程。 - 实现Runnable接口:创建一个实现了
Runnable
接口的类,并实现run()
方法。然后将这个实现类的实例传递给Thread
类的构造函数,并通过新线程的start()
方法来启动。
- 继承Thread类:创建一个新类继承
-
线程的生命周期:新建(New)、就绪(Runnable)、运行(Running)、阻塞(Blocked)和终止(Terminated)。
-
线程的优先级:每个线程都有一个优先级,它们可以从
Thread.MIN_PRIORITY
(1)到Thread.MAX_PRIORITY
(10)变化,Thread.NORM_PRIORITY
(5)是默认优先级。
6.1.2 重点案例:实现一个简单的计数器
假设我们要实现一个简单的计数器,每个线程负责将一个共享变量增加到特定的值。
计数器Runnable实现:
public class Counter implements Runnable {private final int limit;private static int count = 0;public Counter(int limit) {this.limit = limit;}@Overridepublic void run() {while (count < limit) {synchronized (Counter.class) {if (count < limit) {System.out.println(Thread.currentThread().getName() + ": " + (++count));}}}}public static void main(String[] args) {Runnable counter = new Counter(10);new Thread(counter, "Thread-1").start();new Thread(counter, "Thread-2").start();}
}
在这个例子中,我们创建了一个实现了Runnable
接口的Counter
类。每个线程在run()
方法中增加计数器,直到达到了限制值。我们使用synchronized
关键字来确保在同一时刻只有一个线程能够增加计数器。
6.1.3 拓展案例 1:定时器线程
创建一个线程,定时打印消息到控制台,演示如何使用线程来执行定时任务。
import java.util.Timer;
import java.util.TimerTask;public class Reminder {Timer timer;public Reminder(int seconds) {timer = new Timer();timer.schedule(new RemindTask(), seconds * 1000);}class RemindTask extends TimerTask {public void run() {System.out.println("Time's up!");timer.cancel();}}public static void main(String[] args) {new Reminder(5);System.out.println("Task scheduled.");}
}
6.1.4 拓展案例 2:使用 Executor 框架管理线程
Executor框架提供了更高级的接口来管理线程池,使得管理一组任务更加容易。
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;public class ThreadPoolDemo {public static void main(String[] args) {ExecutorService executor = Executors.newFixedThreadPool(5);for (int i = 0; i < 10; i++) {Runnable worker = new WorkerThread("" + i);executor.execute(worker);}executor.shutdown();while (!executor.isTerminated()) {}System.out.println("Finished all threads");}
}class WorkerThread implements Runnable {private String message;public WorkerThread(String message) {this.message = message;}public void run() {System.out.println(Thread.currentThread().getName() + " (Start) message = " + message);processMessage();System.out.println(Thread.currentThread().getName() + " (End)");}private void processMessage() {try {Thread.sleep(2000);} catch (InterruptedException e) {e.printStackTrace();}}
}
通过这些案例,我们看到了Java线程创建和管理的多样化方法。无论是通过实现Runnable
接口,使用定时器任务,还是利用Executor框架管理线程池,正确的线程管理策略都能使你的并发程序运行得更加高效和稳定。现在,带着你的士兵们勇往直前,征服并发编程的世界吧!
6.2 同步机制 - 维持军队的秩序
在并发编程的战场上,正确的同步机制就像是用来维持你的数据军队秩序的军规,确保所有士兵行动协调,避免混乱和冲突。Java提供了多种同步工具和方法,帮助开发者有效管理线程间的协作和资源共享。
6.2.1 基础知识
-
synchronized关键字:可以用于方法或代码块,确保同一时刻只有一个线程可以执行该段代码,从而避免资源冲突或数据不一致的问题。
-
volatile关键字:用于标记变量,确保每次访问变量时都会从主内存中读取,而不是从线程的工作内存,从而保证了变量的可见性。
-
Lock接口:提供了比
synchronized
更灵活的锁定机制,包括可重入锁(ReentrantLock)、读写锁(ReadWriteLock)等,允许更细粒度的锁控制。 -
Condition接口:与Lock配合使用,允许线程间有更细致的通信(比如等待/通知机制),实现线程间的协调。
6.2.2 重点案例:银行转账操作
假设我们需要实现一个银行账户的转账操作,这个操作需要确保线程安全,避免在并发环境下出现资金错误。
public class Account {private int balance;private final Lock lock = new ReentrantLock();public Account(int balance) {this.balance = balance;}// 转账操作public void transfer(Account target, int amount) {lock.lock();try {if (this.balance >= amount) {this.balance -= amount;target.deposit(amount);}} finally {lock.unlock();}}public void deposit(int amount) {lock.lock();try {this.balance += amount;} finally {lock.unlock();}}public int getBalance() {return balance;}
}
在这个例子中,transfer
方法使用了ReentrantLock
来确保转账操作的原子性,避免了并发环境下的资金错误。
6.2.3 拓展案例 1:生产者消费者问题
生产者消费者是并发编程中的一个经典问题,它涉及到两个或多个线程间的协作。使用Lock
和Condition
可以优雅地解决这个问题。
import java.util.LinkedList;
import java.util.Queue;
import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;public class ProducerConsumer {private final Queue<Integer> queue = new LinkedList<>();private final int capacity = 10;private final Lock lock = new ReentrantLock();private final Condition notFull = lock.newCondition();private final Condition notEmpty = lock.newCondition();// 生产者方法public void produce(int value) throws InterruptedException {lock.lock();try {while (queue.size() == capacity) {notFull.await();}queue.add(value);notEmpty.signalAll();} finally {lock.unlock();}}// 消费者方法public int consume() throws InterruptedException {lock.lock();try {while (queue.isEmpty()) {notEmpty.await();}int value = queue.poll();notFull.signalAll();return value;} finally {lock.unlock();}}
}
6.2.4 拓展案例 2:读写锁实现缓存系统
读写锁(ReadWriteLock
)允许多个读操作同时进行,但写操作是互斥的。这对于实现缓存系统来说非常有用。
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;public class Cache {private final Map<String, Object> map = new HashMap<>();private final ReadWriteLock rwLock = new ReentrantReadWriteLock();public Object get(String key) {rwLock.readLock().lock();try {return map.get(key);} finally {rwLock.readLock().unlock();}}public void put(String key, Object value) {rwLock.writeLock().lock();try {map.put(key, value);} finally {rwLock.writeLock().unlock();}}
}
通过这些案例,我们看到了Java中同步机制的强大之处,它不仅帮助我们维持线程间的协作和数据的一致性,还使我们能够设计出高效且线程安全的并发应用。掌握这些同步工具,成为并发编程的指挥官吧!
6.3 并发工具类 - 你的特殊武器
Java的并发工具类就像是隐藏在你的武器库中的特殊武器,它们可以帮助你在并发编程的战场上更加游刃有余。这些工具类提供了强大的功能来帮助管理线程间的协调,以及对共享资源的访问控制,让你能够写出更高效、更健壮的并发程序。
6.3.1 基础知识
-
CountDownLatch:允许一个或多个线程等待其他线程完成一系列操作。当倒计时达到零时,等待的线程被释放继续执行。
-
CyclicBarrier:允许一组线程互相等待,直到所有线程都达到了某个共同点,然后继续执行。
-
Semaphore:一种基于计数的同步机制,可以控制对共享资源的访问。它可以限制同时访问某个特定资源的线程数量。
-
Concurrent Collections:提供了线程安全的集合类,如
ConcurrentHashMap
、CopyOnWriteArrayList
等,用于在并发环境中管理数据。 -
Executor框架:简化了线程的创建和管理,提供了线程池等高级功能,使得并发任务的调度和管理更加灵活和强大。
6.3.2 重点案例:使用 CountDownLatch 协调任务
假设我们有一个任务,需要在开始执行主任务之前,等待其他几个服务初始化完成。
import java.util.concurrent.CountDownLatch;public class ServiceInitializer {private static final int NUM_OF_SERVICES = 3;private final CountDownLatch latch = new CountDownLatch(NUM_OF_SERVICES);public void initialize() {for (int i = 1; i <= NUM_OF_SERVICES; i++) {new Thread(new Service(latch), "Service " + i).start();}try {latch.await(); // 等待所有服务初始化完成System.out.println("All services are initialized. Main task is starting now.");} catch (InterruptedException e) {Thread.currentThread().interrupt();}}static class Service implements Runnable {private final CountDownLatch latch;public Service(CountDownLatch latch) {this.latch = latch;}@Overridepublic void run() {try {// 模拟服务初始化耗时Thread.sleep((long) (Math.random() * 1000));System.out.println(Thread.currentThread().getName() + " initialized.");} catch (InterruptedException e) {Thread.currentThread().interrupt();} finally {latch.countDown();}}}public static void main(String[] args) {new ServiceInitializer().initialize();}
}
6.3.3 拓展案例 1:使用 CyclicBarrier 同步周期性任务
假设我们需要执行一个周期性任务,该任务需要在每个周期内的所有子任务完成后才能开始下一个周期。
import java.util.concurrent.CyclicBarrier;public class CyclicTask implements Runnable {private CyclicBarrier barrier;public CyclicTask(CyclicBarrier barrier) {this.barrier = barrier;}@Overridepublic void run() {try {System.out.println(Thread.currentThread().getName() + " is waiting at the barrier.");barrier.await();System.out.println(Thread.currentThread().getName() + " has crossed the barrier.");} catch (Exception e) {e.printStackTrace();}}public static void main(String[] args) {final int parties = 3;CyclicBarrier barrier = new CyclicBarrier(parties, () -> System.out.println("All parties have arrived at the barrier, let's proceed to the next step."));for (int i = 0; i < parties; i++) {new Thread(new CyclicTask(barrier), "Thread " + (i + 1)).start();}}
}
6.3.4 拓展案例 2:使用 Semaphore 控制资源访问
在某些情况下,我们需要限制对某个资源的并发访问数量
。Semaphore提供了一种简单有效的方法来实现这一目标。
import java.util.concurrent.Semaphore;public class SemaphoreDemo {private static final int MAX_PERMITS = 3;private final Semaphore semaphore = new Semaphore(MAX_PERMITS);public void accessResource() {try {semaphore.acquire();System.out.println(Thread.currentThread().getName() + " is accessing the resource.");Thread.sleep(1000); // 模拟资源访问耗时} catch (InterruptedException e) {e.printStackTrace();} finally {semaphore.release();System.out.println(Thread.currentThread().getName() + " has released the resource.");}}public static void main(String[] args) {SemaphoreDemo demo = new SemaphoreDemo();for (int i = 0; i < 6; i++) {new Thread(demo::accessResource, "Thread " + (i + 1)).start();}}
}
通过这些案例,我们可以看到Java并发工具类如何成为处理并发和多线程问题的强大武器。无论是协调多个任务的完成,同步周期性任务的执行,还是控制对共享资源的访问,这些工具类都能让你的并发编程工作变得更加简单和高效。使用这些特殊的武器,指挥你的数据军团,优雅地完成并发任务!
相关文章:

《Java 简易速速上手小册》第6章:Java 并发编程(2024 最新版)
文章目录 6.1 线程的创建和管理 - 召唤你的士兵6.1.1 基础知识6.1.2 重点案例:实现一个简单的计数器6.1.3 拓展案例 1:定时器线程6.1.4 拓展案例 2:使用 Executor 框架管理线程 6.2 同步机制 - 维持军队的秩序6.2.1 基础知识6.2.2 重点案例&a…...

C++初阶:容器(Containers)list常用接口详解
介绍完了vector类的相关内容后,接下来进入新的篇章,容器list介绍: 文章目录 1.list的初步介绍2.list的定义(constructor)3.list迭代器( iterator )4.string的三种遍历4.1迭代器4.2范围for循环 5…...

HARRYPOTTER: FAWKES
攻击机 192.168.223.128 目标机192.168.223.143 主机发现 nmap -sP 192.168.223.0/24 端口扫描 nmap -sV -p- -A 192.168.223.143 开启了21 22 80 2222 9898 五个端口,其中21端口可以匿名FTP登录,好像有点说法,百度搜索一下发现可以用anonymous登录…...

嵌入式Qt 第一个Qt项目
一.创建Qt项目 打开Qt Creator 界面选择 New Project或者选择菜单栏 【文件】-【新建文件或项目】菜单项 弹出New Project对话框,选择Qt Widgets Application 选择【Choose】按钮,弹出如下对话框 设置项目名称和路径,按照向导进行下一步 选…...
【OpenHarmony硬件操作】风扇与温湿度模块
文章目录 前言一、串行通信是什么二、IC2.1 IC是什么2.2 IC涉及到的线2.3 IC的时序三、风扇的操作3.1 关于 pcf85743.2 风扇的接口函数IO拓展芯片的定义初始化PCF8574初始化 IO拓展版的引脚属性开启和关闭风扇读状态四、温湿度传感器的使用4.1 初始化温湿度传感器</...

Vue3.4+element-plus2.5 + Vite 搭建教程整理
一、 Vue3Vite 项目搭建 说明: Vue3 最新版本已经基于Vite构建,关于Vite简介:Vite 下一代的前端工具链,前端开发与构建工具-CSDN博客 1.安装 并 创建Vue3 应用 npm create vuelatest 创建过程可以一路 NO 目前推荐使用 Vue R…...

STM32Cubmax stm32f103zet6 SPI通讯
一、基本概念 SPI 是英语 Serial Peripheral interface 的缩写,顾名思义就是串行外围设备接口。是 Motorola 首先在其 MC68HCXX 系列处理器上定义的。 SPI 接口主要应用在 EEPROM, FLASH,实时时 钟, AD 转换器,还有数…...
每日OJ题_位运算⑤_力扣371. 两整数之和
目录 力扣371. 两整数之和 解析代码 力扣371. 两整数之和 371. 两整数之和 难度 简单 给你两个整数 a 和 b ,不使用 运算符 和 - ,计算并返回两整数之和。 示例 1: 输入:a 1, b 2 输出:3示例 2: …...
Mysql中索引优化和失效
什么是索引 要了解索引优化和索引失效的场景就要先了解什么是索引 索引是一种有序的存储结构,按照单个或者多个列的值进行排序,以提升搜索效率。 索引的类型 UNIQUE唯一索引 不可以出现相同的值,可以有NULL值。 INDEX普通索引 允许出现相同…...

使用Python+OpenCV2进行图片中的文字分割(支持竖版)
扣字和分割 把图片中的文字,识别出来,并将每个字的图片抠出来; import cv2 import numpy as npHIOG 50 VIOG 3 Position []水平投影 def getHProjection(image):hProjection np.zeros(image.shape,np.uint8)# 获取图像大小(h,w)image.sh…...

Qt中程序发布及常见问题
1、引言 当我们写好一个程序时通常需要发布给用户使用,那么在Qt中程序又是如何实现发布的呢,这里我就来浅谈一下qt中如何发布程序,以及发布程序时的常见问题。 2、发布过程 2.1、切换为release模式 当我们写qt程序时默认是debug模式&#x…...

C语言第二十三弹---指针(七)
✨个人主页: 熬夜学编程的小林 💗系列专栏: 【C语言详解】 【数据结构详解】 指针 1、sizeof和strlen的对比 1.1、sizeof 1.2、strlen 1.3、sizeof 和 strlen的对比 2、数组和指针笔试题解析 2.1、⼀维数组 2.2、二维数组 总结 1、si…...

用HTML5 + JavaScript绘制花、树
用HTML5 JavaScript绘制花、树 <canvas>是一个可以使用脚本 (通常为JavaScript) 来绘制图形的 HTML 元素。 <canvas> 标签/元素只是图形容器,必须使用脚本来绘制图形。 HTML5 canvas 图形标签基础https://blog.csdn.net/cnds123/article/details/112…...

Science重磅_让大模型像婴儿一样学习语言
英文名称: Grounded language acquisition through the eyes and ears of a single child 中文名称: 通过一个孩子的眼睛和耳朵基于实践学习语言 文章: https://www.science.org/doi/10.1126/science.adi1374 代码: https://github.com/wkvong/multimodalbaby 作者: Wai Keen V…...

Java 数据结构篇-实现红黑树的核心方法
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 红黑树的说明 2.0 红黑树的特性 3.0 红黑树的成员变量及其构造方法 4.0 实现红黑树的核心方法 4.1 红黑树内部类的核心方法 (1)判断当前…...

【实战】一、Jest 前端自动化测试框架基础入门(中) —— 前端要学的测试课 从Jest入门到TDD BDD双实战(二)
文章目录 一、Jest 前端自动化测试框架基础入门5.Jest 中的匹配器toBe 匹配器toEqual匹配器toBeNull匹配器toBeUndefined匹配器和toBeDefined匹配器toBeTruthy匹配器toBeFalsy匹配器数字相关的匹配器字符串相关的匹配器数组相关的匹配器异常情况的匹配器 6.Jest 命令行工具的使…...

【C语言 - 力扣 - 反转链表】
反转链表题目描述 给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。 题解1-迭代 假设链表为 1→2→3→∅,我们想要把它改成 ∅←1←2←3。 在遍历链表时,将当前节点的 next 指针改为指向前一个节点。由于节点没…...

ctfshow-php特性(web102-web115)
目录 web102 web103 web104 web105 web106 web107 web108 web109 web110 web111 web112 web113 web114 web115 实践是检验真理的 要多多尝试 web102 <?php highlight_file(__FILE__); $v1$_POST[V1]; $v2$_GET[v2]; $v3$_GET[v3]; $v4is_numeric($v2)and is…...

python系统学习Day1
section1 python introduction 文中tips只做拓展,可跳过。 PartOne introduction 首先要对于python这门语言有一个宏观的认识,包括特点和应用场景。 特点分析: 优势 提供了完善的基础代码库,许多功能不必从零编写简单优雅 劣势 运…...

Idea里自定义封装数据警告解决 Spring Boot Configuration Annotation Processor not configured
我们自定对象封装指定数据,封装类上面一个红色警告,虽然不影响我们的执行,但是有强迫症看着不舒服, 去除方式: 在pom文件加上坐标刷新 <dependency><groupId>org.springframework.boot</groupId><…...
基于算法竞赛的c++编程(28)结构体的进阶应用
结构体的嵌套与复杂数据组织 在C中,结构体可以嵌套使用,形成更复杂的数据结构。例如,可以通过嵌套结构体描述多层级数据关系: struct Address {string city;string street;int zipCode; };struct Employee {string name;int id;…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
Go 语言接口详解
Go 语言接口详解 核心概念 接口定义 在 Go 语言中,接口是一种抽象类型,它定义了一组方法的集合: // 定义接口 type Shape interface {Area() float64Perimeter() float64 } 接口实现 Go 接口的实现是隐式的: // 矩形结构体…...

Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...

Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...
rnn判断string中第一次出现a的下标
# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...

华硕a豆14 Air香氛版,美学与科技的馨香融合
在快节奏的现代生活中,我们渴望一个能激发创想、愉悦感官的工作与生活伙伴,它不仅是冰冷的科技工具,更能触动我们内心深处的细腻情感。正是在这样的期许下,华硕a豆14 Air香氛版翩然而至,它以一种前所未有的方式&#x…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...

JVM 内存结构 详解
内存结构 运行时数据区: Java虚拟机在运行Java程序过程中管理的内存区域。 程序计数器: 线程私有,程序控制流的指示器,分支、循环、跳转、异常处理、线程恢复等基础功能都依赖这个计数器完成。 每个线程都有一个程序计数…...
LangChain知识库管理后端接口:数据库操作详解—— 构建本地知识库系统的基础《二》
这段 Python 代码是一个完整的 知识库数据库操作模块,用于对本地知识库系统中的知识库进行增删改查(CRUD)操作。它基于 SQLAlchemy ORM 框架 和一个自定义的装饰器 with_session 实现数据库会话管理。 📘 一、整体功能概述 该模块…...