Java如何实现多线程场景下的线程安全

简介: 单线程可以正确运行的程序不代表在多线程场景下能够正确运行,这里的正确性往往不容易被发现,它会在并发数达到一定量的时候才可能出现。这也是在测试环节不容易重现的原因。因此,多线程(并发)场景下,如何编写线程安全(Thread-Safety)的程序,对于程序的正确和稳定运行有重要的意义。下面将结合示例,谈谈如何在Java语言中,实现线程安全的程序。

1 引言


    当前随着计算机硬件的快速发展,个人电脑上的CPU也是多核的,现在普遍的CUP核数都是4核或者8核的。因此,在编写程序时,需要为了提高效率,充分发挥硬件的能力,则需要编写并行的程序。Java语言作为互联网应用的主要语言,广泛应用于企业应用程序的开发中,它也是支持多线程Multithreading)的,但多线程虽好,却对程序的编写有较高的要求。

    单线程可以正确运行的程序不代表在多线程场景下能够正确运行,这里的正确性往往不容易被发现,它会在并发数达到一定量的时候才可能出现。这也是在测试环节不容易重现的原因。因此,多线程(并发)场景下,如何编写线程安全(Thread-Safety)的程序,对于程序的正确和稳定运行有重要的意义。下面将结合示例,谈谈如何在Java语言中,实现线程安全的程序。

   为了给出感性的认识,下面给出一个线程不安全的示例,具体如下:

packagecom.example.learn;
publicclassCounter {
privatestaticintcounter=0;
publicstaticintgetCount(){
returncounter;
    }
publicstaticvoidadd(){
counter=counter+1;
    }
}

    这个类有一个静态的属性counter,用于计数。其中可以通过静态方法add()对counter进行加1操作,也可以通过getCount()方法获取到当前的计数counter值。如果是单线程情况下,这个程序是没有问题的,比如循环10次,那么最后获取的计数counter值为10。但多线程情况下,那么这个结果就不一定能够正确获取,可能等于10,也可能小于10,比如9。下面给出一个多线程测试的示例:

packagecom.example.learn;
publicclassMyThreadextendsThread{
privateStringname ;
publicMyThread(Stringname){
this.name=name ;
    }
publicvoidrun(){
Counter.add();
System.out.println("Thead["+this.name+"] Count is "+Counter.getCount());
    }
}
///////////////////////////////////////////////////////////packagecom.example.learn;
publicclassTest01 {
publicstaticvoidmain(String[] args) {
for(inti=0;i<5000;i++){
MyThreadmt1=newMyThread("TCount"+i);
mt1.start();
        }
    }
}

这里为了重现计数的问题,线程数调至比较大,这里是5000。运行此示例,则输出可能结果如下:

Thead[TCount5] Countis4Thead[TCount2] Countis9Thead[TCount4] Countis4Thead[TCount14] Countis10..................................
Thead[TCount4911] Countis4997Thead[TCount4835] Countis4998Thead[TCount4962] Countis4999

   注意:多线程场景下,线程不安全的程序输出结果具有不确定性。

2 synchronized方法


    基于上述的示例,让其变成线程安全的程序,最直接的就是在对应的方法上添加synchronized关键字,让其成为同步的方法。它可以修饰一个类,一个方法和一个代码块。对上述计数程序进行修改,代码如下:

packagecom.example.learn;
publicclassCounter {
privatestaticintcounter=0;
publicstaticintgetCount(){
returncounter;
    }
publicstaticsynchronizedvoidadd(){
counter=counter+1;
    }
}

3 加锁机制


    另外一种常见的同步方法就是加锁,比如Java中有一种重入锁ReentrantLock,它是一种递归无阻塞的同步机制,相对于synchronized来说,它可以提供更加强大和灵活的锁机制,同时可以减少死锁发生的概率。示例代码如下:

packagecom.example.learn;
importjava.util.concurrent.locks.ReentrantLock;
publicclassCounter {
privatestaticintcounter=0;
privatestaticfinalReentrantLocklock=newReentrantLock(true);
publicstaticintgetCount(){
returncounter;
    }
publicstaticvoidadd(){
lock.lock();
try {
counter=counter+1;
        } finally {
lock.unlock();
        }
    }
}

4 使用Atomic对象


    由于锁机制会影响一定的性能,而有些场景下,可以通过无锁方式进行实现。Java内置了Atomic相关原子操作类,比如AtomicInteger, AtomicLong, AtomicBoolean和AtomicReference,可以根据不同的场景进行选择。下面给出示例代码:

packagecom.example.learn;
importjava.util.concurrent.atomic.AtomicInteger;
publicclassCounter {
privatestaticfinalAtomicIntegercounter=newAtomicInteger();
publicstaticintgetCount(){
returncounter.get();
    }
publicstaticvoidadd(){
counter.incrementAndGet();
    }
}

5 无状态对象


    前面提到,线程不安全的一个原因就是多个线程同时访问某个对象中的数据,数据存在共享的情况,因此,如果将数据变成独享的,即无状态(stateless)的话,那么自然就是线程安全的。而所谓的无状态的方法,就是给同样的输入,就能返回一致的结果。下面给出示例代码:

packagecom.example.learn;
publicclassCounter {
publicstaticintsum (intn) {
intret=0;
for (inti=1; i<=n; i++) {
ret+=i;
        }
returnret;
    }
}

    前面提到了几种线程安全的方法,总体的思想要不就是通过锁机制实现同步,要不就是防止数据共享,防止在多个线程中对数据进行读写操作。另外,有些文章中说到,可以在变量前使用volatile修饰,来实现同步机制,但这个经过测试是不一定的,有些场景下,volatile依旧不能保证线程安全。虽然上述是线程安全的经验总结,但是还是需要通过严格的测试进行验证,实践是检验真理的唯一标准。

相关文章
|
1天前
|
Java
【深度解读】Java条件语句:if-else与switch的适用场景大起底!
【6月更文挑战第14天】本文探讨了Java中if-else和switch条件语句的适用场景。if-else以其灵活性处理复杂逻辑判断,适合多种条件组合;而switch在多分支选择上更高效简洁,尤其适用于枚举类型和固定常量。了解两者特点有助于写出更优雅、高效的代码。
|
1天前
|
监控 Java API
Java 程序设计 第八章 线程
Java 程序设计 第八章 线程
|
1天前
|
存储 安全 Java
Java多线程编程--JUC
Java多线程编程
|
2天前
|
缓存 NoSQL Java
Java高并发实战:利用线程池和Redis实现高效数据入库
Java高并发实战:利用线程池和Redis实现高效数据入库
12 0
|
3天前
|
Java API
详细探究Java多线程的线程状态变化
Java多线程的线程状态主要有六种:新建(NEW)、可运行(RUNNABLE)、阻塞(BLOCKED)、等待(WAITING)、超时等待(TIMED_WAITING)和终止(TERMINATED)。线程创建后处于NEW状态,调用start()后进入RUNNABLE状态,表示准备好运行。当线程获得CPU资源,开始执行run()方法时,它处于运行状态。线程可以因等待锁或调用sleep()等方法进入BLOCKED或等待状态。线程完成任务或发生异常后,会进入TERMINATED状态。
|
3天前
|
存储 安全 Java
Java多线程中线程安全问题
Java多线程中的线程安全问题主要涉及多线程环境下对共享资源的访问可能导致的数据损坏或不一致。线程安全的核心在于确保在多线程调度顺序不确定的情况下,代码的执行结果始终正确。常见原因包括线程调度随机性、共享数据修改以及原子性问题。解决线程安全问题通常需要采用同步机制,如使用synchronized关键字或Lock接口,以确保同一时间只有一个线程能够访问特定资源,从而保持数据的一致性和正确性。
|
3天前
|
安全 Java API
Java并发基础-启动和终止线程
Java并发基础-启动和终止线程
13 0
|
3天前
|
Java 调度
Java并发基础-线程简介(状态、常用方法)
Java并发基础-线程简介(状态、常用方法)
8 0
|
8天前
|
安全 Java 编译器
JAVA-多线程知识点总结(二)
JAVA-多线程知识点总结(二)
|
3天前
|
监控 安全 Java
Java多线程的使用
Java多线程允许程序同时执行多个任务,提高了系统的整体性能和响应速度。通过创建Thread类或其子类的实例,或使用Runnable接口,Java开发者可以定义并发执行的代码段。多线程在处理复杂任务、资源共享、网络通信等方面具有显著优势,但也需要注意线程安全、同步和死锁等问题。Java提供了丰富的API和工具来处理这些并发问题,使多线程编程更加高效和可靠。