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

目录
  • 前言
  • 无状态
  • 没有共享状态
  • 消息传递
  • 不可变状态
  • 使用来自 java.util.concurrent 的数据结构
  • 同步块
  • 易失性领域
  • 总结

前言

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

(0)

相关推荐

  • Java8新特性之线程安全日期类

    LocalDateTime Java8新特性之一,新增日期类. 在项目开发过程中经常遇到时间处理,但是你真的用对了吗,理解阿里巴巴开发手册中禁用static修饰SimpleDateFormat吗 通过阅读本篇文章你将了解到: 为什么需要LocalDate.LocalTime.LocalDateTime[java8新提供的类] Java8新的时间API的使用方式,包括创建.格式化.解析.计算.修改 可以使用Instant代替 Date,LocalDateTime代替 Calendar,DateTi

  • 如何测试Java类的线程安全性

    这篇文章主要介绍了如何测试Java类的线程安全性,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 线程安全性是Java等语言/平台中类的一个重要标准,在Java中,我们经常在线程之间共享对象.由于缺乏线程安全性而导致的问题很难调试,因为它们是偶发的,而且几乎不可能有目的地重现.如何测试对象以确保它们是线程安全的? 假如有一个内存书架 package com.mzc.common.thread; import java.util.Map; impo

  • Java 集合中的类关于线程安全

    Java集合中那些类是线程安全的 线程安全类 在集合框架中,有些类是线程安全的,这些都是jdk1.1中的出现的.在jdk1.2之后,就出现许许多多非线程安全的类. 下面是这些线程安全的同步的类: vector:就比arraylist多了个同步化机制(线程安全),因为效率较低,现在已经不太建议使用.在web应用中,特别是前台页面,往往效率(页面响应速度)是优先考虑的. statck:堆栈类,先进后出 hashtable:就比hashmap多了个线程安全 enumeration:枚举,相当于迭代器

  • Java线程安全的常用类_动力节点Java学院整理

    线程安全类 在集合框架中,有些类是线程安全的,这些都是jdk1.1中的出现的.在jdk1.2之后,就出现许许多多非线程安全的类. 下面是这些线程安全的同步的类: vector:就比arraylist多了个同步化机制(线程安全),因为效率较低,现在已经不太建议使用.在web应用中,特别是前台页面,往往效率(页面响应速度)是优先考虑的. statck:堆栈类,先进后出 hashtable:就比hashmap多了个线程安全 除了这些之外,其他的集合大都是非线程安全的类和接口. 线程安全的类其方法是同步

  • Java多线程环境下SimpleDateFormat类安全转换

    一.SimpleDateFormat类 package state; import java.text.ParseException; import java.text.SimpleDateFormat; import java.util.Date; /** * SimpleDateFormat类负责日期的转换与格式化 * 解决SimpleDateFormat类多线程环境下转换错误问题 * @author zc * */ public class SimpleDateFormatThread e

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

    目录 前言 无状态 没有共享状态 消息传递 不可变状态 使用来自 java.util.concurrent 的数据结构 同步块 易失性领域 总结 前言 几乎每个 Java 应用程序都使用线程.像 Tomcat 这样的 Web 服务器在单独的工作线程中处理每个请求,胖客户端在专用工作线程中处理长时间运行的请求,甚至批处理使用 java.util.concurrent.ForkJoinPool 来提高性能. 因此,有必要以线程安全的方式编写类,这可以通过以下技术之一来实现. 无状态 当多个线程访问同

  • Java线程关闭的3种方法

    Java线程关闭,总的来说有3种: 1.使用状态位,这个简单,就不多说了: 复制代码 代码如下: public class Task extends Thread { private volatile boolean flag= true; public void stopTask() { flag = false; } @Override public void run() { while(flag){ /* do your no-block task */ } } } 2.当线程等待某些事件

  • Java线程休眠的5种方法

    目录 方法1:Thread.sleep 方法2:TimeUnit 方法3:wait 方法4:Condition 方法5:LockSupport 总结 前言: 在 Java 中,让线程休眠的方法有很多,这些方法大致可以分为两类,一类是设置时间,在一段时间后自动唤醒,而另一个类是提供了一对休眠和唤醒的方法,在线程休眠之后,可以在任意时间对线程进行唤醒. PS:休眠是指让某个线程暂停执行(进入等待状态),唤醒指的是让某个暂停的线程继续执行. 线程休眠的方法有以下 5 个: Thread.sleep T

  • java编程创建型设计模式单例模式的七种示例

    目录 1.什么是单例模式? 2.七种写法 2.1饿汉式(静态常量) 2.2饿汉式(静态代码块) 2.3懒汉式(线程不安全) 2.4懒汉式(线程安全,同步方法) 2.5双重校验锁 2.6静态内部类 2.7枚举 3.单例模式在JDK中的应用(简单的源码分析) 4.单例模式总结 1.什么是单例模式? 所谓类的单例设计模式,就是采取一定的方法保证在整个的软件系统中,对某个类只能存在一个对象实例,并且该类只提供一个取得其对象实例的方法(静态方法). 比如Hibernate的 SessionFactory,

  • Java线程池的几种实现方法及常见问题解答

    工作中,经常会涉及到线程.比如有些任务,经常会交与线程去异步执行.抑或服务端程序为每个请求单独建立一个线程处理任务.线程之外的,比如我们用的数据库连接.这些创建销毁或者打开关闭的操作,非常影响系统性能.所以,"池"的用处就凸显出来了. 1. 为什么要使用线程池 在3.6.1节介绍的实现方式中,对每个客户都分配一个新的工作线程.当工作线程与客户通信结束,这个线程就被销毁.这种实现方式有以下不足之处: •服务器创建和销毁工作的开销( 包括所花费的时间和系统资源 )很大.这一项不用解释,可以

  • Java之线程编程的4种方法实现案例讲解

    1.继承Thread public class T4 { public static void main(String[] args) { System.out.println(Thread.currentThread()); Thread t1 = new A1(); t1.start(); } } class A1 extends Thread{ @Override public void run() { for(int i=0;i<10;i++) { System.out.println(

  • Java线程池的几种实现方法和区别介绍实例详解

    下面通过实例代码为大家介绍Java线程池的几种实现方法和区别: import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.List; import java.util.Random; import java.util.concurrent.Callable; import java.util.

  • Java线程池的几种实现方法和区别介绍

    Java线程池的几种实现方法和区别介绍 import java.text.DateFormat; import java.text.SimpleDateFormat; import java.util.ArrayList; import java.util.Date; import java.util.List; import java.util.Random; import java.util.concurrent.Callable; import java.util.concurrent.E

  • java线程池的四种创建方式详细分析

    目录 前言 1. 线程池 2. 创建方式 前言 在讲述线程池的前提 先补充一下连接池的定义 连接池是创建和管理一个连接的缓冲池的技术,这些连接准备好被任何需要它们的线程使用 可以看到其连接池的作用如下: 1. 线程池 线程池(英语:thread pool):一种线程使用模式.线程过多会带来调度开销,进而影响缓存局部性和整体性能.而线程池维护着多个线程,等待着监督管理者分配可并发执行的任务.这避免了在处理短时间任务时创建与销毁线程的代价.线程池不仅能够保证内核的充分利用,还能防止过分调度 特点:

  • 详解mysql插入数据后返回自增ID的七种方法

    引言 mysql 和 oracle 插入的时候有一个很大的区别是: oracle 支持序列做 id: mysql 本身有一个列可以做自增长字段. mysql 在插入一条数据后,如何能获得到这个自增 id 的值呢? 一:使用 last_insert_id() SELECT LAST_INSERT_ID(); 1. 每次 mysql 的 query 操作在 mysql 服务器上可以理解为一次"原子"操作, 写操作常常需要锁表, 这里的锁表是 mysql 应用服务器锁表不是我们的应用程序锁表

随机推荐