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

 更新时间:2022年06月11日 09:42:27   作者:学习Java中  
线程安全是指某个方法或某段代码,在多线程中能够正确的执行,不会出现数据不一致或数据污染的情况,我们把这样的程序称之为线程安全的,反之则为非线程安全的,下面这篇文章主要给大家介绍了关于创建Java线程安全类的七种方法,需要的朋友可以参考下

前言

几乎每个 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线程安全类创建内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java中两种抛出异常的方式示例详解

    Java中两种抛出异常的方式示例详解

    在Java中有两种抛出异常的方式,一种是throw,直接抛出异常,另一种是throws,间接抛出异常,本文给大家详细说明java中两种抛出异常的方式,感兴趣的朋友一起看看吧
    2023-08-08
  • jmeter设置全局变量与正则表达式提取器过程图解

    jmeter设置全局变量与正则表达式提取器过程图解

    这篇文章主要介绍了jmeter设置全局变量与正则表达式提取器过程图解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-10-10
  • spring boot+自定义 AOP 实现全局校验的实例代码

    spring boot+自定义 AOP 实现全局校验的实例代码

    最近公司重构项目,重构为最热的微服务框架 spring boot, 重构的时候遇到几个可以统一处理的问题。这篇文章主要介绍了spring boot+自定义 AOP 实现全局校验 ,需要的朋友可以参考下
    2019-04-04
  • 图文浅析Java序列化和反序列化

    图文浅析Java序列化和反序列化

    序列化(Serialization)是将对象的状态信息转化为可以存储或者传输的形式的过程,下面这篇文章主要给大家介绍了关于Java序列化和反序列化的相关资料,需要的朋友可以参考下
    2021-05-05
  • Java编程调用微信接口实现图文信息推送功能

    Java编程调用微信接口实现图文信息推送功能

    这篇文章主要介绍了Java编程调用微信接口实现图文信息等推送功能,涉及java微信图文信息推送接口调用及相关文件、字符串编码转换相关操作技巧,需要的朋友可以参考下
    2017-08-08
  • mybatis-plus动态表名的实现示例

    mybatis-plus动态表名的实现示例

    这篇文章主要介绍了mybatis-plus动态表名的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • mybatis批量插入时,有字段可能为null会报错问题

    mybatis批量插入时,有字段可能为null会报错问题

    这篇文章主要介绍了mybatis批量插入时,有字段可能为null会报错问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-11-11
  • Java日常练习题,每天进步一点点(22)

    Java日常练习题,每天进步一点点(22)

    下面小编就为大家带来一篇Java基础的几道练习题(分享)。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧,希望可以帮到你
    2021-07-07
  • IDEA + Maven环境下的SSM框架整合及搭建过程

    IDEA + Maven环境下的SSM框架整合及搭建过程

    这篇文章主要介绍了IDEA + Maven环境下的SSM框架整合及搭建过程,本文通过图文并茂的形式给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-01-01
  • Java动态代理简单介绍

    Java动态代理简单介绍

    动态代理指的是,代理类和目标类的关系在程序运行的时候确定的,客户通过代理类来调用目标对象的方法,是在程序运行时根据需要动态的创建目标类的代理对象。本文将通过案例详细讲解一下Java动态代理的原理及实现,需要的可以参考一下
    2022-08-08

最新评论