`

java多线程的陷阱

    博客分类:
  • java
阅读更多

1、在构造函数中启动线程

我在很多代码中都看到这样的问题,在构造函数中启动一个线程,类似这样:

  1. public class A{  
  2.    public A(){  
  3.       this.x=1;  
  4.       this.y=2;  
  5.       this.thread=new MyThread();  
  6.       this.thread.start();  
  7.    }  
  8.      
  9. }    

这个会引起什么问题呢?如果有个类B继承了类A,依据java类初始化的顺序,A的构造函数一定会在B的构造函数调用前被调用,那么thread线程也将在B被完全初始化之前启动,当thread运行时使用到了类A中的某些变量,那么就可能使用的不是你预期中的值,因为在B的构造函数中你可能赋给这些变量新的值。也就是说此时将有两个线程在使用这些变量,而这些变量却没有同步。

解决这个问题有两个办法:将A设置为final,不可继承;或者提供单独的start方法用来启动线程,而不是放在构造函数中。

2、不完全的同步

都知道对一个变量同步的有效方式是用synchronized保护起来,synchronized可能是对象锁,也可能是类锁,看你是类方法还是实例方法。但是,当你将某个变量在A方法中同步,那么在变量出现的其他地方,你也需要同步,除非你允许弱可见性甚至产生错误值。类似这样的代码:

  1. class A{  
  2.   int x;  
  3.   public int getX(){  
  4.      return x;  
  5.   }  
  6.   public synchronized void setX(int x)  
  7.   {  
  8.      this.x=x;  
  9.   }  
  10. }     

x的setter方法有同步,然而getter方法却没有,那么就无法保证其他线程通过getX得到的x是最新的值。事实上,这里的setX的同步是没有必要的,因为对int的写入是原子的,这一点JVM规范已经保证,多个同步没有任何意义;当然,如果这里不是int,而是double或者long,那么getX和setX都将需要同步,因为double和long都是64位,写入和读取都是分成两个32位来进行(这一点取决于jvm的实现,有的jvm实现可能保证对long和double的read、write是原子的),没有保证原子性。类似上面这样的代码,其实都可以通过声明变量为volatile来解决。

3、在使用某个对象当锁时,改变了对象的引用,导致同步失效。

这也是很常见的错误,类似下面的代码:

  1. synchronized(array[0])  
  2. {  
  3.    ......  
  4.    array[0]=new A();  
  5.    ......  
  6. }    

同步块使用array[0]作为锁,然而在同步块中却改变了array[0]指向的引用。分析下这个场景,第一个线程获取了array[0]的锁,第二个线程因为无法获取array[0]而等待,在改变了array[0]的引用后,第三个线程获取了新的array[0]的锁,第一和第三两个线程持有的锁是不一样的,同步互斥的目的就完全没有达到了。这样代码的修改,通常是将锁声明为final变量或者引入业务无关的锁对象,保证在同步块内不会被修改引用。

4、没有在循环中调用wait()。

wait和notify用于实现条件变量,你可能知道需要在同步块中调用wait和notify,为了保证条件的改变能做到原子性和可见性。常常看见很多代码做到了同步,却没有在循环中调用wait,而是使用if甚至没有条件判断:

  1. synchronized(lock)  
  2. {  
  3.    if(isEmpty()  
  4.      lock.wait();  
  5.      
  6. }  
  7.      

对条件的判断是使用if,这会造成什么问题呢?在判断条件之前可能调用notify或者notifyAll,那么条件已经满足,不会等待,这没什么问题。在条件没有满足,调用了wait()方法,释放lock锁并进入等待休眠状态。如果线程是在正常情况下,也就是条件被改变之后被唤醒,那么没有任何问题,条件满足继续执行下面的逻辑操作。问题在于线程可能被意外甚至恶意唤醒,由于没有再次进行条件判断,在条件没有被满足的情况下,线程执行了后续的操作。意外唤醒的情况,可能是调用了notifyAll,可能是有人恶意唤醒,也可能是很少情况下的自动苏醒(称为“伪唤醒”)。因此为了防止这种条件没有满足就执行后续操作的情况,需要在被唤醒后再次判断条件,如果条件不满足,继续进入等待状态,条件满足,才进行后续操作。

  1. synchronized(lock)  
  2. {  
  3.    while(isEmpty()  
  4.      lock.wait();  
  5.      
  6. }     

没有进行条件判断就调用wait的情况更严重,因为在等待之前可能notify已经被调用,那么在调用了wait之后进入等待休眠状态后就无法保证线程苏醒过来。

5、同步的范围过小或者过大。

同步的范围过小,可能完全没有达到同步的目的;同步的范围过大,可能会影响性能。同步范围过小的一个常见例子是误认为两个同步的方法一起调用也是将同步的,需要记住的是Atomic+Atomic!=Atomic。

  1. Map map=Collections.synchronizedMap(new HashMap());  
  2. if(!map.containsKey("a")){  
  3.          map.put("a", value);  
  4. }    

这是一个很典型的错误,map是线程安全的,containskey和put方法也是线程安全的,然而两个线程安全的方法被组合调用就不一定是线程安全的了。因为在containsKey和put之间,可能有其他线程抢先put进了a,那么就可能覆盖了其他线程设置的值,导致值的丢失。解决这一问题的方法就是扩大同步范围,因为对象锁是可重入的,因此在线程安全方法之上再同步相同的锁对象不会有问题。

  1. Map map = Collections.synchronizedMap(new HashMap());  
  2. synchronized (map) {  
  3.      if (!map.containsKey("a")) {  
  4.          map.put("a", value);  
  5.      }  
  6.  }    

注意,加大锁的范围,也要保证使用的是同一个锁,不然很可能造成死锁。 Collections.synchronizedMap(new HashMap())使用的锁是map本身,因此没有问题。当然,上面的情况现在更推荐使用ConcurrentHashMap,它有putIfAbsent方法来达到同样的目的并且满足线程安全性。

同步范围过大的例子也很多,比如在同步块中new大对象,或者调用费时的IO操作(操作数据库,webservice等)。不得不调用费时操作的时候,一定要指定超时时间,例如通过URLConnection去invoke某个URL时就要设置connect timeout和read timeout,防止锁被独占不释放。同步范围过大的情况下,要在保证线程安全的前提下,将不必要同步的操作从同步块中移出。

6、正确使用volatile

在jdk5修正了volatile的语义后,volatile作为一种轻量级的同步策略就得到了大量的使用。volatile的严格定义参考jvm spec,这里只从volatile能做什么,和不能用来做什么出发做个探讨。

volatile可以用来做什么?

1)状态标志,模拟控制机制。常见用途如控制线程是否停止:

  1. private volatile boolean stopped;  
  2. public void close(){  
  3.    stopped=true;  
  4. }  
  5.  
  6. public void run(){  
  7.  
  8.    while(!stopped){  
  9.       //do something  
  10.    }  
  11.      

前提是do something中不会有阻塞调用之类。volatile保证stopped变量的可见性,run方法中读取stopped变量总是main memory中的最新值。

2)安全发布,如修复DLC问题。

  1. private volatile IoBufferAllocator instance;  
  2. public IoBufferAllocator getInsntace(){  
  3.     if(instance==null){  
  4.         synchronized (IoBufferAllocator.class) {  
  5.             if(instance==null)  
  6.                 instance=new IoBufferAllocator();  
  7.         }  
  8.     }  
  9.     return instance;  

3)开销较低的读写锁

  1. public class CheesyCounter {  
  2.     private volatile int value;  
  3.  
  4.     public int getValue() { return value; }  
  5.  
  6.     public synchronized int increment() {  
  7.         return value++;  
  8.     }  
  9. }  

synchronized保证更新的原子性,volatile保证线程间的可见性。

volatile不能用于做什么?

1)不能用于做计数器

  1. public class CheesyCounter {  
  2.     private volatile int value;  
  3.  
  4.     public int getValue() { return value; }  
  5.  
  6.     public int increment() {  
  7.         return value++;  
  8.     }  

因为value++其实是有三个操作组成的:读取、修改、写入,volatile不能保证这个序列是原子的。对value的修改操作依赖于value的最新值。解决这个问题的方法可以将increment方法同步,或者使用AtomicInteger原子类。

2)与其他变量构成不变式

一个典型的例子是定义一个数据范围,需要保证约束lower< upper。

  1. public class NumberRange {  
  2.     private volatile int lower, upper;  
  3.  
  4.     public int getLower() { return lower; }  
  5.     public int getUpper() { return upper; }  
  6.  
  7.     public void setLower(int value) {   
  8.         if (value > upper)   
  9.             throw new IllegalArgumentException();  
  10.         lower = value;  
  11.     }  
  12.  
  13.     public void setUpper(int value) {   
  14.         if (value < lower)   
  15.             throw new IllegalArgumentException();  
  16.         upper = value;  
  17.     }  
  18. }  

尽管讲lower和upper声明为volatile,但是setLower和setUpper并不是线程安全方法。假设初始状态为(0,5),同时调用setLower(4)和setUpper(3),两个线程交叉进行,最后结果可能是(4,3),违反了约束条件。修改这个问题的办法就是将setLower和setUpper同步:

  1. public class NumberRange {  
  2.     private volatile int lower, upper;  
  3.  
  4.     public int getLower() { return lower; }  
  5.     public int getUpper() { return upper; }  
  6.  
  7.     public synchronized void setLower(int value) {   
  8.         if (value > upper)   
  9.             throw new IllegalArgumentException();  
  10.         lower = value;  
  11.     }  
  12.  
  13.     public synchronized void setUpper(int value) {   
  14.         if (value < lower)   
  15.             throw new IllegalArgumentException();  
  16.         upper = value;  
  17.     }  
分享到:
评论

相关推荐

    汪文君JAVA多线程编程实战(完整不加密)

    学习《汪文君JAVA多线程编程实战》不仅能够提高读者对Java多线程编程的理解,还有助于培养良好的并发编程习惯,避免常见的并发陷阱。对于想要提升自己在并发编程领域技能的Java开发者来说,这本书无疑是一份宝贵的...

    Java多线程知识,龙果学院

    "Java多线程知识,龙果学院"这一课程显然是针对这部分内容进行深入讲解的资源,旨在帮助开发者提升在多任务环境下的编程能力。 一、Java多线程基础 1. **线程的概念**:线程是程序执行的最小单元,一个进程可以有...

    java 多线程应用

    使用多线程时遇到的问题,并提出那些常见陷阱的解决方案。

    Java多线程编程详解:核心概念与高级技术应用

    内容概要:本文详细介绍了 Java多线程编程的基础知识和高级技术。内容涵盖线程的概念与重要性、创建线程的方式、线程的生命周期与基本控制方法、线程同步与死锁、线程间通信、线程池与 Executor框架、并发集合与原子...

    Java多线程设计模式

    Java多线程设计模式是程序开发中的重要领域,特别是在服务器端和高性能应用中,多线程的使用可以显著提升系统的并行处理能力。本书由知名技术作者结城浩撰写,于2005年出版,得到了广泛的好评,被誉为“好书”。这...

    java多线程设计模式(pdf)

    《Java多线程设计模式》是一本专注于Java并发编程的权威指南,对于深入理解Java并发机制和提升多线程编程技能具有极高的价值。这本书详细介绍了如何在Java环境中有效地使用多线程,以及如何设计出高效、可维护的并发...

    java多线程资料

    《Java多线程资料》包含了两个重要的资源,一本是Addison.Wesley.Java.Concurrency.in.Practice.May.2006.chm,另一本是[大家网]Java Threads, Third Edition[www.TopSage.com].chm。这两本书籍都是深入理解Java多...

    JAVA 的多线程浅析.pdf

    ### JAVA的多线程浅析 #### 一、JAVA语言的背景与特点 Java自诞生以来,便以其独特的魅力在编程领域占据了一席之地。它由Sun Microsystems开发,以其跨平台性、安全性、面向对象特性以及多线程能力著称。在互联网...

    java陷阱常见面试题

    5. 在多线程环境下,如何正确使用synchronized关键字? 6. Java中如何实现线程间的通信? 7. 解释Java中的单例模式及其可能的陷阱。 8. 如何在Java中处理网络异常,防止资源泄露? 9. 请解释JVM的类加载过程及双亲...

    电子书《java线程》

    本电子书详细介绍了如何在Java环境中有效地使用多线程,优化程序性能,并避免并发编程中的常见陷阱。 在Java中,线程是程序执行的独立路径,它们可以同时运行,使程序能够处理多个任务。Java提供了一个丰富的线程...

    JAVA多线程编程详解-详细操作例子

    本主题将深入探讨“JAVA多线程编程详解-详细操作例子”,结合提供的资源,我们可以从以下几个方面进行学习: 1. **线程的基本概念**: 线程是程序执行的最小单位,一个进程可以有多个线程。在Java中,可以通过实现...

    汪文君高并发编程实战视频资源下载.txt

    │ 高并发编程第一阶段05讲、采用多线程方式模拟银行排队叫号.mp4 │ 高并发编程第一阶段06讲、用Runnable接口将线程的逻辑执行单元从控制中抽取出来.mp4 │ 高并发编程第一阶段07讲、策略模式在Thread和Runnable...

    java 多线程

    总结来说,Java多线程是一个深度广泛的领域,涵盖了许多概念和工具,从基本的线程创建到复杂的并发控制和同步机制。理解和熟练掌握这些知识对于编写高效、健壮的多线程Java应用程序至关重要。《Java Concurrency in ...

    基于多线程的网络爬虫设计与实现.pdf

    ### 基于多线程的网络爬虫设计与实现 #### 概述 网络爬虫作为一种高效的数据抓取工具,在大数据时代扮演着极其重要的角色。通过对互联网网页内容的自动检索与下载,网络爬虫为数据挖掘、搜索引擎优化等工作提供了强...

    java面试中的陷阱java面试中的陷阱

    Hashtable是线程安全的,因此在多线程环境下使用更为合适。 #### 四、断言(assert)的使用 - 断言是一种程序调试工具,用于验证程序中的假设条件是否成立。如果断言表达式的值为false,则会抛出AssertionError...

    Java Scjp 陷阱大全

    `synchronized`关键字、`volatile`变量和`java.util.concurrent`包的使用是多线程陷阱的重点。 5. **类与对象**:类的继承、封装和多态是面向对象编程的基础。理解构造函数、访问修饰符和抽象类的使用,以及`final`...

    外文文献-编写多线程的 Java 应用程序.doc

    这篇外文文献旨在探讨多线程编程中的问题,并提出解决常见陷阱的方法。 首先,我们需要理解什么是线程。一个程序或进程可以包含多个线程,这些线程按照程序代码执行指令。就像一台计算机上可以运行多个进程一样,多...

    429.427.JAVA基础教程_多线程-复习:线程的创建与常用方法(429).rar

    本教程聚焦于Java多线程的创建与常用方法,帮助开发者深入理解这一关键概念。 在Java中,线程的创建主要有两种方式:继承Thread类和实现Runnable接口。继承Thread类是最直接的方式,只需要重写run()方法,然后创建...

    Java面试资料-高清完整PDF版(集合、IO、多线程、反射前端知识、框架).rar

    本压缩包提供的"Java面试资料-高清完整PDF版(集合、IO、多线程、反射前端知识、框架)"是一个全面的复习资源,涵盖了Java开发中的关键知识点。 首先,我们来详细探讨一下这些知识点: 1. **面向对象基础**:面向...

Global site tag (gtag.js) - Google Analytics