热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

(三)java并发编程线程的安全性

(1)什么是线程安全?当一个类被多个线程访问的时,这个类始终能表现出正确的行为,那么就称这个类是线程安全的。(2)什么是有状态对象

(1)什么是线程安全?

当一个类被多个线程访问的时,这个类始终能表现出正确的行为,那么就称这个类是线程安全的。

 

(2)什么是有状态对象,什么是无状态对象呢?无状态对象一定是线程安全的。

有状态对象,就是有数据存储功能的对象。有状态对象,就是有示例变量的对象,可以保存数据,是非线程安全的。

无状态对象,就是不能保存数据。就是没有成员变量的对象,不能保存数据,是线程安全的。

 

(3)安全与不安全

不安全就是存在的数据是多个线程共享的才不安全,而每个线程自己独有的局部变量,则是安全的。所以可以总结如下:

1)常量始终是线程安全的,因为只存在只读操作。

2)每次调用方法前都新建一个实例是线程安全的,因为不会访问共享资源(共享堆资源)。

3)局部变量是安全的。因为每执行一个方法,都会独立的空间创建局部变量,它不是共享资源。局部变量包括方法的参数变量和方法的内部变量。

3.1)举个栗子

线程安全的调用方式,每次都new一个新的对象,每个对象在自己的堆空间中,多个线程不共享。

 

   public classSafeThreadClient {

      publicstatic voidmain(String[]args) {

 

          for(inti&#61;0;i<10000;i&#43;&#43;){

           UnsafeSequenceunsafeSequence &#61;newUnsafeSequence();

           Threadthread &#61;newThread(unsafeSequence);

           thread.start();

           }

 

       }

    }

线程安全不安全调用&#xff0c;如下&#xff0c;new一次&#xff0c;多个线程共享堆中的这一个对象的成员变量。

   public classUnsafeThreadClient {

      publicstatic voidmain(String[]args) {

           UnsafeSequenceunsafeSequence &#61;newUnsafeSequence();

          for(inti&#61;0;i<10000;i&#43;&#43;){

           Threadthread &#61;newThread(unsafeSequence);

           thread.start();

           }

 

       }

    }

   

   public classUnsafeSequenceimplementsRunnable{

      privateintvalue;

      publicintgetNext(){

      returnvalue&#43;&#43;;

       }

      publicvoidrun(){

           System.out.println(getNext());

       }

 

    }

 

3.2)上述代码内存分析&#xff1a;

非线程安全&#xff1a;


 

线程安全&#xff1a;


就像上面的例子&#xff0c;value每次都要加1&#xff0c;但是可能会出现这样的问题&#xff0c;当两个线程同时到达&#xff08;读取--修改--写入&#xff09;&#xff0c;同时读取value&#xff0c;同时对value&#43;1&#xff0c;然后写入到内存中&#xff0c;假如刚才的操作value4&#xff0c;然后两个线程同时操作&#xff0c;value的值应该加两次&#xff0c;但可能两个线程在读value值的时候都读到的是4&#xff0c;都会加1&#xff0c;写入内存都会认为是5。 而实际上两个线程对value应该加了两次&#xff0c;此时的value6才对&#xff0c;这就是在多线程情况下&#xff0c;这个类表现的某些错误行为&#xff0c;我们就说这个类是线程不安全的。

 

而究其结果是因为&#xff0c;在 读出--修改--写入这个过程中&#xff0c;其中的一个状态都依赖于上一个状态。

最终会导致数据的不完整性&#xff0c;或不确定性&#xff0c;就是在不恰当的执行顺序&#xff0c;导致不正确的结果&#xff0c;有一个名字就是线程交替执行的时候&#xff0c;就会发生竞争条件。

 

(4)竞争条件&#xff1f;

当某个计算的正确性取决于多个线程的交替时序时&#xff0c;那么就会发生竞争条件。

(5)复合操作

要避免竞争条件问题&#xff0c;就必须在某个线程修改该变量的时候&#xff0c;通过某个方式阻止其他线程使用这个变量&#xff0c;从而确保其他线程只能在修改操作完成之前或者之后读取和修改状态&#xff0c;而不是在修改状态中。

加入A B两个线程&#xff0c;从A执行角度来看&#xff0c;B执行的时候&#xff0c;要么B全部执行完&#xff0c;要么完全不执行B。那么AB的执行来说是原子的。原子操作是&#xff0c;对于访问同一个状态的所有操作&#xff08;包括操作本身&#xff09;这个操作是以原子方式执行的操作。

 

(5.1)利用java线程安全原子类AtomicLong

利用线程安全类来确保vaue在读取--修改--写入的一致性。修改代码如下所示&#xff1a;

 

package unsafeThread;

importjava.util.concurrent.atomic.AtomicLong;

 

/**

* Created by fang on 2017/11/18.

*/

   public classSafeSequenceimplementsRunnable{

      privatefinalAtomicLongvalue&#61;newAtomicLong(0);

 

      publicintgetNext(){

          value.incrementAndGet();

 

          return(int)value.get();

       }

 

      publicvoidrun(){

       System.out.println(getNext());

    }

 

}

 

packageunsafeThread;

 

/**

* Created by fang on 2017/11/18.

*/

   public classSafeThreadClient {

   public static voidmain(String[]args) {

 

   //利用AtomicLongjava.util.corrurnent包中的,线程安全类计数器.

    SafeSequence safeSequence &#61; new SafeSequence();

      for(inti&#61;0;i<10000;i&#43;&#43;){

           Threadthread &#61;newThread(safeSequence);

           thread.start();

           }

       }

    }

(5.2)内存分析&#xff1a;

 


java.util.concurrent.atomicjava封装好的原子性的类包&#xff0c;在这里我们用线程安全的类AtomicLong来管理对象的属性变量&#xff0c;所以这个类是安全的。&#xff08;看jdk8的源码可以看到AtomicLong实现线程安全的基本原理是CAS-compare and swap&#xff0c;具体就不在这里写了&#xff0c;后续文章详细讲解jdk源码&#xff09;

 

(5.3)利用java内置锁

java提供了一种内置机制来支持原子性:同步代码块&#xff0c;代码如下所示。

 

   public classUnsafeSequenceimplementsRunnable{

      privateintvalue;

      publicsynchronized intgetNext(){

          returnvalue&#43;&#43;;

       }

      publicvoidrun(){

           System.out.println(getNext());

       }

 

    }

我们在方法上添加上synchronized&#xff0c;被synchronized锁定的块就会以原子性的方式执行。

 

(6)重入

尽管synchronized锁住的不是同一个方法块,但是做到一个线程同时调这两个方法。

例如如下代码&#xff1a;

packagesafeThread;

 

/**

 * Created by fang on 2017/11/19.

 */

public classWidget {

   public synchronized voiddoSomething(){

       System.out.println("i&#39;m father, in execute...");

   }

}

 

 

packagesafeThread;

 

/**

 * Created by fang on 2017/11/19.

 */

public classLoggingWidget extends Widget {

   public synchronized voiddoSomething(){

       System.out.println("child " &#43; toString() &#43; ": callingdoSomething");

       super.doSomething();

   }

}

 

packagesafeThread;

 

/**

 * Created by fang on 2017/11/19.

 * Synchronized可重复锁,举例.

 */

public classReentrantSynchronized {

   /**

    * main主线程.

    * &#64;param args

    */

   public static voidmain(String[] args) {

       LoggingWidget loggingWidget &#61; new LoggingWidget();

       loggingWidget.doSomething();

 

   }

}

 

内存分析


就像家里的大门有一把锁,屋子里也有锁,对于屋子里的锁来说&#xff0c;我是同一个“线程”&#xff0c;既然有进入大门的权限&#xff0c;就一定有进入屋子的权限。

(7)活跃性与性能

一些保证线程安全的方法&#xff0c;线程安全保证了&#xff0c;但是程序的执行效率可能降低了。因为&#xff0c;当访问量很大的时候&#xff0c;如果是多cpu或者cpu多核的时候&#xff0c;请求只能一个一个的被执行&#xff0c;需要等待前一个完全执行才可以执行&#xff0c;可能会影响程序效率。

当使用锁的时候&#xff0c;我们应该清楚代码块中的实现功能&#xff0c;以及在执行该代码块时是否需要很长时间。无论是执行密度操作还是执行某个可能的阻塞操作&#xff0c;如果持有锁的时间过长&#xff0c;那么就可能带来活跃性或者性能问题。

 

(8)总结

了解到了线程安全和非安全线程&#xff0c;对象和人一样都要有“安全感”&#xff0c;下一篇对象的共享。

 

 


推荐阅读
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • Python 异步编程:深入理解 asyncio 库(上)
    本文介绍了 Python 3.4 版本引入的标准库 asyncio,该库为异步 IO 提供了强大的支持。我们将探讨为什么需要 asyncio,以及它如何简化并发编程的复杂性,并详细介绍其核心概念和使用方法。 ... [详细]
  • 本文详细探讨了KMP算法中next数组的构建及其应用,重点分析了未改良和改良后的next数组在字符串匹配中的作用。通过具体实例和代码实现,帮助读者更好地理解KMP算法的核心原理。 ... [详细]
  • 本文详细介绍了 GWT 中 PopupPanel 类的 onKeyDownPreview 方法,提供了多个代码示例及应用场景,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 探讨如何通过编程技术实现100个并发连接,解决线程创建顺序问题,并提供高效的并发测试方案。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • 本文详细介绍如何使用Python进行配置文件的读写操作,涵盖常见的配置文件格式(如INI、JSON、TOML和YAML),并提供具体的代码示例。 ... [详细]
  • 本文将介绍如何使用 Go 语言编写和运行一个简单的“Hello, World!”程序。内容涵盖开发环境配置、代码结构解析及执行步骤。 ... [详细]
  • Java 类成员初始化顺序与数组创建
    本文探讨了Java中类成员的初始化顺序、静态引入、可变参数以及finalize方法的应用。通过具体的代码示例,详细解释了这些概念及其在实际编程中的使用。 ... [详细]
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • 主要用了2个类来实现的,话不多说,直接看运行结果,然后在奉上源代码1.Index.javaimportjava.awt.Color;im ... [详细]
author-avatar
mongcheng
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有