iis服务器助手广告广告
返回顶部
首页 > 资讯 > 后端开发 > Python >创建Java线程安全类的七种方法
  • 950
分享到

创建Java线程安全类的七种方法

2024-04-02 19:04:59 950人浏览 独家记忆

Python 官方文档:入门教程 => 点击学习

摘要

目录前言无状态没有共享状态消息传递不可变状态使用来自 java.util.concurrent 的数据结构同步块易失性领域总结前言 几乎每个 Java 应用程序都使用线程。像 Tom

前言

几乎每个 Java 应用程序都使用线程。像 Tomcat 这样的 WEB 服务器在单独的工作线程中处理每个请求,胖客户端在专用工作线程中处理长时间运行的请求,甚至批处理使用 java.util.concurrent.ForkJoinPool 来提高性能。

因此,有必要以线程安全的方式编写类,这可以通过以下技术之一来实现。

无状态

当多个线程访问同一个实例或静态变量时,您必须以某种方式协调对该变量的访问。最简单的方法就是避免使用实例或静态变量。没有实例变量的类中的方法只使用局部变量和方法参数。下面的例子展示了这样一个方法,它是类 java.lang.Math 的一部分:

public static int subtractExact(int x, int y) {
    int r = x - y;
    if (((x ^ y) & (x ^ r)) < 0) {
        throw new ArithmeticException("integer overflow");
    }
    return r;
}

没有共享状态

如果您无法避免状态,请不要共享状态。状态应该只由单个线程拥有。这种技术的一个例子是 SWT 或 Swing 图形用户界面框架的事件处理线程。

您可以通过扩展线程类并添加实例变量来实现线程局部实例变量。在以下示例中,字段 pool 和 workQueue 对于单个工作线程是本地的。

package java.util.concurrent;
public class ForkJoinWorkerThread extends Thread {
    final ForkJoinPool pool;                
    final ForkJoinPool.WorkQueue workQueue; 
}

实现线程局部变量的另一种方法是将类 java.lang.ThreadLocal 用于要使线程局部的字段。下面是一个使用 java.lang.ThreadLocal 的实例变量示例:

public class CallbackState {
public static final ThreadLocal<CallbackStatePerThread> callbackStatePerThread = 
    new ThreadLocal<CallbackStatePerThread>()
   {
      @Override
        protected CallbackStatePerThread  initialValue()
      { 
       return getOrCreateCallbackStatePerThread();
      }
   };
}

您将实例变量的类型包装在 java.lang.ThreadLocal 中。您可以通过方法 initialValue() 为您的 java.lang.ThreadLocal 提供初始值。

下面展示了如何使用实例变量:

CallbackStatePerThread callbackStatePerThread = CallbackState.callbackStatePerThread.get();

通过调用 get() 方法,您会收到与当前线程关联的对象。

由于在应用程序服务器中,使用许多线程池来处理请求,因此 java.lang.ThreadLocal 会导致此环境中的内存消耗很高。因此,不建议将 java.lang.ThreadLocal 用于由应用程序服务器的请求处理线程执行的类。

消息传递

如果您不使用上述技术共享状态,则需要一种线程进行通信的方式。做到这一点的一种技术是在线程之间传递消息。您可以使用 java.util.concurrent 包中的并发队列实现消息传递。或者,更好的是,使用Akka 之类的框架,这是一个演员风格并发的框架。以下示例显示了如何使用 Akka 发送消息:

target.tell(message, getSelf());

并收到一条消息:

@Override
public Receive createReceive() {
     return receiveBuilder()
        .match(String.class, s -> System.out.println(s.toLowerCase()))
        .build();
}

不可变状态

为了避免发送线程在另一个线程读取消息时更改消息的问题,消息应该是不可变的。因此,Akka 框架的约定是所有消息都必须是不可变的

当你实现一个不可变类时,你应该将它的字段声明为 final。这不仅可以确保编译器可以检查这些字段实际上是不可变的,而且即使它们被错误地发布,也可以使它们正确初始化。这是最终实例变量的示例:

public class ExampleFinalField
{
    private final int finalField;
    public ExampleFinalField(int value)
    {
        this.finalField = value;
    }
}

使用来自 java.util.concurrent 的数据结构

消息传递使用并发队列进行线程之间的通信。并发队列是 java.util.concurrent 包中提供的数据结构之一。这个包提供了并发映射、队列、出队、集合和列表的类。这些数据结构经过高度优化和线程安全测试

同步块

如果您不能使用上述技术之一,请使用同步。通过将锁放在同步块中,您可以确保一次只有一个线程可以执行此部分。

synchronized(lock)
{
    i++;
}

请注意,当您使用多个嵌套同步块时,可能会出现死锁。当两个线程试图获取另一个线程持有的锁时,就会发生死锁。

易失性领域

正常的非易失性字段可以缓存在寄存器或缓存中。通过将变量声明为 volatile,您可以告诉JVM和编译器始终返回最新写入的值。这不仅适用于变量本身,还适用于线程写入 volatile 字段的所有值。下面显示了一个 volatile 实例变量的示例:

public class ExampleVolatileField
{
    private volatile int  volatileField;
}

如果写入不依赖于当前值,您可以使用 volatile 字段。或者,如果您可以确保一次只有一个线程可以更新该字段。

总结

到此这篇关于创建Java线程安全类的七种方法的文章就介绍到这了,更多相关Java线程安全类创建内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: 创建Java线程安全类的七种方法

本文链接: https://www.lsjlt.com/news/151137.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 创建Java线程安全类的七种方法
    目录前言无状态没有共享状态消息传递不可变状态使用来自 java.util.concurrent 的数据结构同步块易失性领域总结前言 几乎每个 Java 应用程序都使用线程。像 Tom...
    99+
    2024-04-02
  • Java创建线程的七种方法,全网最全面总结~
    目录 前言 一、继承Thread,重写run方法 二、实现Runnable接口,重写run方法 三、使用匿名内部类创建 Thread 子类对象 四、使用匿名内部类,实现Runnable接口 五、lambda表达式 六、实现Callable...
    99+
    2023-09-13
    java 多线程
  • java创建多线程的七种方式
    一、继承Thread,重写run方法 通过自定义一个类(这里起名为:MyThread),继承Thread类,重写run方法,最后在main方法中new出MyThread实例,调用这个实例的继承的Thread类的start方法创建一个线程。 ...
    99+
    2023-09-26
    java
  • java创建线程池一共有七种方式
    java创建线程池一共有七种方式 这 7 种实现方法分别是: Executors.newFixedThreadPool:创建一个固定大小的线程池,可控制并发的线程数,超出的线程会在队列中等待。 Exe...
    99+
    2023-09-13
    java 开发语言
  • 如何创建Java线程安全类
    今天小编给大家分享一下如何创建Java线程安全类的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。无状态当多个线程访问同一个实例...
    99+
    2023-07-02
  • java创建线程的两种方法区别
    在Java中创建一个线程有两种方法:继承Thread类和实现Runnable接口。下面通过两个例子来分析两者的区别:1)继承Thread类public class TestThread extends Thread { int count...
    99+
    2023-05-31
    java 创建线程 ava
  • Java线程的三种创建方式
    目录1、Thread2、Runnable和Thread3、Runnable和Thread4、三者对比5、注意项1、Thread 继承Thread类,并重写run方法 class ...
    99+
    2024-04-02
  • Java线程的创建方法
    这篇文章主要讲解了“Java线程的创建方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java线程的创建方法”吧!多线程指的是一个程序运行时,会包含多个线程同时进行。Java创建线程有三种...
    99+
    2023-06-02
  • Java 中创建线程的几种方式
    Java 是一种面向对象的编程语言,它支持多线程编程。多线程编程是指在一个程序中同时运行多个线程,这些线程可以并行执行,以提高程序的效率和性能。Java 提供了多种创建线程的方法,本文将介绍这些方法以...
    99+
    2023-09-13
    java jvm servlet
  • Java线程创建与Thread类的使用方法
    目录1.线程与Thread类1.1操作系统中的线程与Java线程1.1.1线程与Thread类1.1.2Thread类的构造方法1.1.3启用java线程必会的方法1.2第一个Jav...
    99+
    2024-04-02
  • 一文搞懂Java创建线程的五种方法
    目录题目描述解题思路代码详解第一种 继承Thread类创建线程第二种:实现Runnable接口创建线程第三种:实现Callable接口,通过FutureTask包装器来创建Threa...
    99+
    2024-04-02
  • java多线程创建及线程安全详解
    什么是线程 线程被称为轻量级进程,是程序执行的最小单位,它是指在程序执行过程中,能够执行代码的一个执行单位。每个程序程序都至少有一个线程,也即是程序本身。 线程...
    99+
    2024-04-02
  • 创建线程的三种基本方法
    这篇文章主要讲解了“创建线程的三种基本方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“创建线程的三种基本方法”吧!挺基础的知识,一开始不是很愿意写,毕竟这...
    99+
    2024-04-02
  • Java线程创建的四种方式总结
    多线程的创建,方式一:继承于Thread类 1.创建一个继承于Thread类的子类 2.重写Thread类的run()--->将此线程执行的操作声明在run()中 3.创建Th...
    99+
    2024-04-02
  • 聊聊java多线程创建方式及线程安全问题
    什么是线程 线程被称为轻量级进程,是程序执行的最小单位,它是指在程序执行过程中,能够执行代码的一个执行单位。每个程序程序都至少有一个线程,也即是程序本身。 线程的状态 新建(New)...
    99+
    2024-04-02
  • 详解Java线程池的使用(7种创建方法)
    目录 1. 固定数量的线程池a.  线程池返回结果b. ⾃定义线程池名称或优先级2. 带缓存的线程池3. 执⾏定时任务 a.&nbs...
    99+
    2023-03-24
    Java线程池 Java线程池使用 线程池
  • Java创建多线程的8种方式集合
    目录1、继承Thread类,重写run()方法2、实现Runnable接口,重写run()3、匿名内部类的方式4、带返回值的线程(实现implements Callable<返...
    99+
    2024-04-02
  • Java的线程与进程以及线程的四种创建方式
    目录问题描述case 代码截图数据库DOcontroller定义dao定义mapper实现mysql相关 properties配置数据库数据测试结果具体错误信息解决总结问题描述 这里...
    99+
    2024-04-02
  • java创建线程的方法有哪些
    java中创建线程的方法有:1.使用Runnable接口创建;2.使用Thread继承类创建;3.使用Callable和Future创建;java中创建线程的方法有以下几种使用Runnable接口创建public class Runnabl...
    99+
    2024-04-02
  • Java匿名内部类创建线程的方法是什么
    在Java中,可以使用匿名内部类创建线程的方法是通过继承Thread类或实现Runnable接口。1. 继承Thread类:```T...
    99+
    2023-09-11
    Java
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作