Java中的基础构建模块(第五章)

简介:

Java中的基础构建模块

Java平台类库包含了丰富的并发基础构建模块,例如线程安全的容器类以及各种用于协调多个相互协作的线程控制流的同步工具类。

1.同步容器类

同步容器类都是线程安全的,但在某些情况下可能需要额外的客户端加锁来保护复合操作。常见的复合操作包括:迭代、跳转(在容器内元素之间)、条件运算(例如“若没有则添加”)。

隐式迭代:某些情况下迭代操作会隐藏起来。如下代码中println调用Set的toString方法,然后对Set中的对象进行迭代调用toString方法:

public class HiddenIterator {
    private final Set<Integer> set = new HashSet<Integer>();
    public void addTenThings() {
        Random r = new Random();
        for (int i = 0;i < 10;i++) add(r.nextInt());
        System.out.println("Debug : " + set);
    }
}

2.并发容器类

并发容器专门针对多个线程并发访问设计。
通过使用并发容器代替同步容器,可以极大提高伸缩性并降低风险。

  1. ConcurrentHashMap:替代同步的基于散列Map,在其接口中增加了一些常见符合操作的支持,如“若没有则添加”、替换有条件删除等。

        ConcurrentHashMap使用了一种粒度更细的加锁机制:分段锁来实现更大程度的共享,能够在并发环境下提高吞吐量。

    ConcurrentHashMap不能被加锁来执行独占访问。

在实际使用中,只有当应用程序需要加锁以进行独占访问时,才应该放弃使用ConcurrentHashMap。

ConcurrentHashMap中的原子操作:

方法 说明
v putIfAbsent(K key,V value) 仅当k没有相应的映射值时才插入
boolean remove(K key,V value) 仅当k被映射到v时才移除
boolean replace(K key,V oldValue,V newValue) 仅当k被映射到oldValue时才进行替换
V replace(K key,V newValue) 仅当k被映射到某个值时才进行替换
  1. CopyOnWriteArrayLIst:代替同步List,提供了更好的并发性能,并且在迭代期间不需要对容器进行加锁或复制。
    “写入时复制(Copy-On-Write)”容器的线程安全性在于,只要正确地发布一个事实不可变对象,那么在访问该对象时就不需要进一步的控制,在每次修改时,都会创建并发布一个新的容器副本,从而实现可变性。

    仅当迭代操作远远多于修改操作时,才应该使用“写入时复制”容器。
    
  2. BlockingQueue(生产者----消费者):相对于Queue,增加了可阻塞的插入和获取等操作。如果队列为空,那么获取元素的操作将一直阻塞,直到队列中出现一个可用的元素,如果队列已满(对于有界队列),那么插入操作将会一直阻塞,直到队列中出现可用空间。
    阻塞队列提供了可阻塞的take和put方法,以及支持定时的offer和poll方法。

    在构建高可靠的应用程序时,有界队列是一种强大的资源管理工具:它们能抑制并产生过多的工作项,使应用程序在负荷过载的情况下变得更加健壮。所以应该通过阻塞队列在设计中构建资源管理机制。
    
    • LinkedBlockingQueue
    • ArrayBlockingQueue
    • PriorityBlockingQueue

串行线程封闭:线程封闭对象只能由一个线程拥有,但可以通过安全地发布该对象来“转移”所有权,在转移所有权后,也只有另一个线程能获得这个对象的访问权限,并且发布对象的线程不会再访问它。

  1. 双端队列
    Deque---->BlockingDeque---->ArrayBlockingQueue、LinkedBlockingQueue

3.阻塞与中断

线程阻塞的原因:

- 等待I/O操作结束
- 等待获得一个锁
- 等待从sleep方法醒来
- 等待另一个线程的计算结果

Thread.interrupt()用于中断线程。Thread.interrupt()方法不会中断一个正在运行的线程。这一方法实际上完成的是,在线程受到阻塞时抛出一个中断信号,这样线程就得以退出阻塞的状态。更确切的说,如果线程被Object.wait, Thread.join和 Thread.sleep三种方法之一阻塞,那么,它将接收到一个中断异常(InterruptedException),从而提早地终结被阻塞状态。

因此,如果线程被上述几种方法阻塞,正确的停止线程方式是设置共享变量,并调用interrupt()(注意变量应该先设置)。如果线程没有被阻塞,这时调用interrupt()将不起作用;否则,线程就将得到异常(该线程必须事先预备好处理此状况),接着逃离阻塞状态。在任何一种情况中,最后线程都将检查共享变量然后再停止。

4.同步工具类

同步工具类可以是任何一个对象,只要它根据其自身的状态来协调线程的控制流。阻塞队列可以作为同步工具类,其他类型的同步工具类还包括信号量(Semaphore)栅栏(Barrier)以及闭锁(Latch)

  • 闭锁(Latch):闭锁是一种同步工具类,可以延迟线程的进度直到其到达终止状态。闭锁的作用相当于一扇门:在闭锁到达结束状态之前,这扇门一直是关闭的,并且没有任何线程能够通过,当到达结束状态时,这扇门会打开并允许所有的线程通过。当闭锁到达结束状态后,将不会再改变状态,因此这扇门将永远保持打开状态。
    闭锁可以用来确保某些活动直到其他活动都完成后才继续执行,如:

    1. 确保某个计算在其需要的所有资源都被初始化之后才继续执行
    2. 确保某个服务在其依赖的所有其他服务都已经启动之后才启动
    3. 等待直到某个操作的所有参与者(如在多玩家游戏中的所有玩家)都就绪再继续执行。

Java提供了闭锁的实现:CountDownLatch。它可以使一个或多个线程等待一组时间发生。闭锁状态包括一个计数器,该计数器被初始化为一个正数,表示需要等待的事件数量。countDown()方法递减计数器,表示有一个时间已经发生了;await()方法等待计数器到达0,表示所有需要等待的事件都已经发生,如果计数器非0,await会一直阻塞直到计数器为0,或等待中的线程中断、超时。

  • FutureTask
    FutureTask实现了Future语义,表示一种抽象的可生成结果的计算。Future.get的行为取决于任务的状态,如果任务已经完成,那么get会立即返回结果,否则get将阻塞直到任务进入完成状态,然后返回结果或者抛出异常。

FutureTask表示的计算是通过Callable来实现的,相当于一种可生成结果的Runnable,并且可以处于以下3种状态:等待运行(Waiting to run)、正在运行(Running)和运行完成(Completed)。
*关于FutureTask的详细信息可以参看blog:
Java中创建线程的方法*

  • 信号量
    计数信号量(Counting Semaphore)用来控制同时访问某个特定资源的操作数量,或者同时执行某个指定操作的数量,计数信号量还可以用来实现某种资源池,或者对容器施加边界。
  • 栅栏
    栅栏类似于闭锁,它能则色一组线程直到某个事件发生。栅栏与闭锁的关键区别在于:所有线程必须同时到达栅栏位置,才能继续执行。闭锁用于等待事件,而栅栏用于等待其他线程。如果所有线程都达到了栅栏位置,那么栅栏将打开,此时所有的线程都被释放,而栅栏将被重置以便下次使用。

Java中提供CyclicBarrier实现栅栏功能。

相关文章
|
2月前
|
设计模式 消息中间件 传感器
Java 设计模式之观察者模式:构建松耦合的事件响应系统
观察者模式是Java中常用的行为型设计模式,用于构建松耦合的事件响应系统。当一个对象状态改变时,所有依赖它的观察者将自动收到通知并更新。该模式通过抽象耦合实现发布-订阅机制,广泛应用于GUI事件处理、消息通知、数据监控等场景,具有良好的可扩展性和维护性。
289 8
|
2月前
|
机器学习/深度学习 人工智能 自然语言处理
Java与生成式AI:构建内容生成与创意辅助系统
生成式AI正在重塑内容创作、软件开发和创意设计的方式。本文深入探讨如何在Java生态中构建支持文本、图像、代码等多种生成任务的创意辅助系统。我们将完整展示集成大型生成模型(如GPT、Stable Diffusion)、处理生成任务队列、优化生成结果以及构建企业级生成式AI应用的全流程,为Java开发者提供构建下一代创意辅助系统的完整技术方案。
222 10
|
2月前
|
人工智能 算法 Java
Java与AI驱动区块链:构建智能合约与去中心化AI应用
区块链技术和人工智能的融合正在开创去中心化智能应用的新纪元。本文深入探讨如何使用Java构建AI驱动的区块链应用,涵盖智能合约开发、去中心化AI模型训练与推理、数据隐私保护以及通证经济激励等核心主题。我们将完整展示从区块链基础集成、智能合约编写、AI模型上链到去中心化应用(DApp)开发的全流程,为构建下一代可信、透明的智能去中心化系统提供完整技术方案。
274 3
|
2月前
|
机器学习/深度学习 人工智能 监控
Java与AI模型部署:构建企业级模型服务与生命周期管理平台
随着企业AI模型数量的快速增长,模型部署与生命周期管理成为确保AI应用稳定运行的关键。本文深入探讨如何使用Java生态构建一个企业级的模型服务平台,实现模型的版本控制、A/B测试、灰度发布、监控与回滚。通过集成Spring Boot、Kubernetes、MLflow和监控工具,我们将展示如何构建一个高可用、可扩展的模型服务架构,为大规模AI应用提供坚实的运维基础。
271 0
|
2月前
|
人工智能 Java 物联网
Java与边缘AI:构建离线智能的物联网与移动应用
随着边缘计算和终端设备算力的飞速发展,AI推理正从云端向边缘端迁移。本文深入探讨如何在资源受限的边缘设备上使用Java构建离线智能应用,涵盖从模型优化、推理加速到资源管理的全流程。我们将完整展示在Android设备、嵌入式系统和IoT网关中部署轻量级AI模型的技术方案,为构建真正实时、隐私安全的边缘智能应用提供完整实践指南。
345 3
|
2月前
|
人工智能 监控 Java
Java与AI智能体:构建自主决策与工具调用的智能系统
随着AI智能体技术的快速发展,构建能够自主理解任务、制定计划并执行复杂操作的智能系统已成为新的技术前沿。本文深入探讨如何在Java生态中构建具备工具调用、记忆管理和自主决策能力的AI智能体系统。我们将完整展示从智能体架构设计、工具生态系统、记忆机制到多智能体协作的全流程,为Java开发者提供构建下一代自主智能系统的完整技术方案。
433 4
|
2月前
|
机器学习/深度学习 分布式计算 Java
Java与图神经网络:构建企业级知识图谱与智能推理系统
图神经网络(GNN)作为处理非欧几里得数据的前沿技术,正成为企业知识管理和智能推理的核心引擎。本文深入探讨如何在Java生态中构建基于GNN的知识图谱系统,涵盖从图数据建模、GNN模型集成、分布式图计算到实时推理的全流程。通过具体的代码实现和架构设计,展示如何将先进的图神经网络技术融入传统Java企业应用,为构建下一代智能决策系统提供完整解决方案。
358 0
|
2月前
|
人工智能 缓存 自然语言处理
Java与多模态AI:构建支持文本、图像和音频的智能应用
随着大模型从单一文本处理向多模态能力演进,现代AI应用需要同时处理文本、图像、音频等多种信息形式。本文深入探讨如何在Java生态中构建支持多模态AI能力的智能应用。我们将完整展示集成视觉模型、语音模型和语言模型的实践方案,涵盖从文件预处理、多模态推理到结果融合的全流程,为Java开发者打开通往下一代多模态AI应用的大门。
358 41
|
3月前
|
人工智能 Java API
Java AI智能体实战:使用LangChain4j构建能使用工具的AI助手
随着AI技术的发展,AI智能体(Agent)能够通过使用工具来执行复杂任务,从而大幅扩展其能力边界。本文介绍如何在Java中使用LangChain4j框架构建一个能够使用外部工具的AI智能体。我们将通过一个具体示例——一个能获取天气信息和执行数学计算的AI助手,详细讲解如何定义工具、创建智能体并处理执行流程。本文包含完整的代码示例和架构说明,帮助Java开发者快速上手AI智能体的开发。
1166 8
|
3月前
|
人工智能 缓存 监控
使用LangChain4j构建Java AI智能体:让大模型学会使用工具
AI智能体是大模型技术的重要演进方向,它使模型能够主动使用工具、与环境交互,以完成复杂任务。本文详细介绍如何在Java应用中,借助LangChain4j框架构建一个具备工具使用能力的AI智能体。我们将创建一个能够进行数学计算和实时信息查询的智能体,涵盖工具定义、智能体组装、记忆管理以及Spring Boot集成等关键步骤,并展示如何通过简单的对话界面与智能体交互。
1035 1