`
luciesfly
  • 浏览: 70685 次
  • 性别: Icon_minigender_1
  • 来自: 上海
文章分类
社区版块
存档分类
最新评论

java线程同步的解决方案

阅读更多
1 如何同步保证线程安全

1、 问题描述:

  如果一个资源或对象可能被多个线程同时访问,它就是一个共享资源;例如类的成员变量,包括类变量和实例变量,再比如对一个文件进行写操作等。一般情况下,对共享资源的访问需要考虑线程安全的问题。

  如果一个对象的完整生命周期只在一个线程内,则不需要考虑线程安全,例如一个局部变量。下面为一个示例代码:

   1. public class C1 {
   2. public static java.text.SimpleDateFormat sdf = new java.text.SimpleDateFormat("yyyy-MM-dd");
   3. //其他代码
   4. }

  假如在一个JSP中这样的去调用:

   1. <a.jsp>:
   2. <%
   3. Java.util.Date date = C1.sdf.parse(“2003-4-15”);
   4. %>

  则这样的代码不是线程安全的。因为java.text.SimpleDateFormat不是线程安全的,a.jsp中的代码将会有若干个线程同时执行,而都访问的是同一个线程不安全的对象,这样就不是一个线程安全的代码。正确的写法应该如下:

   1. <a.jsp>:
   2. <%
   3. java.text.SimpleDateFormat sdf = new java.text.SimpleDateFormat("yyyy-MM-dd");
   4. Java.util.Date date = sdf.parse(“2003-4-15”);
   5. %>

2、 原因分析:

  此时,sdf对象从创建到销毁都位于一个方法中,相当于一个局部变量,不是一个共享资源,因此则没有线程不安全的问题。

3、 解决方法或过程:

1) 如果对象是immutable,则是线程安全的,例如:String,可以放心使用。
2) 如果对象是线程安全的,则放心使用
3) 有条件线程安全,对于Vector和Hashtable一般情况下是线程安全的,但是对于某些特殊情况,需要通过额外的synchronized保证线程安全。
4) 使用synchronized关键字;

  对于上例中可以改写jsp代码,在sdf上进行同步,而不需要每次创建一个新的对象来保证线程安全,代码如下:

   1. <%
   2. synchronized(C1.sdf){
   3. Java.util.Date date = C1.sdf.parse(“2003-4-15”);
   4. }
   5. %>

  这种写法是在一个对象级别上进行同步,也就是说任何时候,对于这个对象,最多只能有一个线程在执行同步方法。

  另外一种写法是在Class级别上进行同步,写法如下:

   1. public class C1 {
   2. public static java.text.SimpleDateFormat sdf = new java.text.SimpleDateFormat("yyyy-MM-dd");
   3. public void method(){
   4. synchronized(C1.class){
   5. //synchronized code
   6. }
   7. }
   8. }

  这种写法表示无论C1有多少个实例,在任何一个时间点,最多只能有一个线程和一个实例进入同步块中。这种同步会比较大的影响性能。

  5) 有些对象不能在多线程间共享,则只能在方法内部使用,或者只在一个线程内部使用。



前言

如果开发者正开发或维护基于Servlet的Web应用,则Servlet规范建议最好能够看看。因为它含有的内容对于Web应用开发者理解Servlet容器的工作机理很有帮助。

其中,规范给出了Servlet容器是如何处理客户请求的。Servlet容器将会根据web.xml配置文件中定义的各个Servet而创建相应的单例。因此,多个客户请求可能同时访问这些单例,即多个线程同时访问它们。在Web应用中保证线程安全是很重要的。开发者应该对这个问题保持警惕,而且必须确保各自的代码必须以线程安全的方式运行。

温习线程安全

大部分Java开发者都应该听过synchronized关键字。在不采用任何第三方库的前提下,Java本身对线程提供了原生支持,而且synchronized关键字往往是Java应用中实现线程安全最重要的因素。Java中的同步提供了互斥支持。通过同步一块代码或整个方法能够保证同时最多只有单个线程执行它,从而实现了线程安全。引入同步具有副作用,即阻塞。比如,大公司或律师办公室的前台小姐同时需要处理电话、邮件、受访客户等等。这使得她的工作很繁忙,而且导致一些事情不能够及时处理。

在Web应用中需要警惕阻塞。受同步保护的代码块使得其同时处理客户请求的吞吐量降低,而且很多客户处于阻塞状态,除非某客户处理完成。而且互斥不仅会带来阻塞,还会带来死锁。通常,死锁是不可恢复的。如下条件将触发死锁的发生:线程A锁住了线程B等待的资源,而且线程B锁住了线程A等待的资源,即线程B一直在等待线程A释放锁,线程A也是如此。因此,对于多线程的应用而言,死锁的预防和处理通常都是很头疼的。

另外,synchronized关键字还使得大量的同步对象到处使用,从而引入了死锁的可能性。比如,java.util.Hashtable和java.util.Vector中提供的方法都是受互斥保护的,因此除非确实需要使用它们,否则尽量不用。开发者只需要使用java.util.HashMap和java.util.ArrayList即可。当然,java.util.Collections中的同步方法也使用了synchronized关键字。

尽管可重入更易于管理,但它引入了其他问题。可重入代码避免了线程间数据的共享。考虑如下代码(姑且认为Java中的方法是线程安全的):



 public Double pi() {   int a = 22;   int b = 7;   return new Double(a / b);   }

不管同时进入该方法的线程有多少,它总是线程安全的。各个线程都维护了属于各个线程的栈,并不同其他线程共享。其中,各个线程在当前方法(包括静态方法)中创建的方法变量仅属于当前线程,即存储在当前线程的栈中。因此,当线程A和B同时进入上述方法时,它们都将创建a和b。由于上述方法不存在数据共享,因此上述方法是线程安全的。请注意:22/7值同PI值较接近,但它们不相等。

接下来,看看如何优化上述代码吧。



 private Double pi = null;   public Double pi() {   if (pi == null) {   pi = new Double(22 / 7);   }   return pi;   }

尽管改进后的方法能够提高性能,但并不是线程安全的。比如:如果pi为null,而且线程 A和B同时进入第4行。因此,线程A和B会同时测试pi是否为空,它们都将返回true。接下来,如果线程A继续执行(线程B由于某种原因被暂挂),然后返回对内存地址的引用。其中,该内存地址含有22/7的结果,即pi值。最后,线程A退出方法。当线程B再次进入第5行时,新的内存地址将覆盖原先的内存地址(线程A提供的)。这太危险了,而且这种问题往往难于调试。

如果使用 ThreadLocal,则不仅能够保证pi()方法是线程安全,而且能够提供性能的改善。



private static ThreadLocal pi = new ThreadLocal();   public Double pi() {   if (pi.get() == null) {   pi.set(new Double(22 / 7));   }   return (Double)pi.get();   }

ThreadLocal类能够包裹任何对象,而且能够将对象绑定到当前线程,使得它仅仅供当前线程使用。当线程初次执行pi()方法时,由于没有对象绑定到ThreadLocal实例pi上,因此get()方法返回null。借助于set() 方法能够将对象绑定到当前线程,而且不供其它线程使用。因此,如果不同线程需要经常访问pi()方法,则借助于ThreadLocal不仅能够保证线程安全,而且能够提高性能。

目前,存在很多关于如何使用ThreadLocal的资源。在Java 1.4之前,ThreadLocal的性能确实很差,但是现已解决了这个问题。另外,由于对ThreadLocal的错误理解,使得很多开发者对它的误用。注意,上述实例使用ThreadLocal的方式是绝对没问题的。在引入ThreadLocal后,上述方法的行为并未发生改变,但是方法已经是线程安全的了。

通过可重入的方式开发线程安全的代码要求开发者谨慎使用实例变量或静态变量,尤其对于修改那些其他线程需要使用的对象而言。某些场合,使用同步可能更为合适。然而,为识别由于同步而引起的应用性能瓶颈往往只能借助于专业的性能评测工具或负载测试完成。

Web应用中的线程安全

在温习线程安全的知识后,来研究Web应用中是如何线程安全的吧!开发者通过创建Web页面来操作数据库。比如,在Web层和业务逻辑层都能够操作RDBMS。本文使用Hibernate将业务模型持久化到数据库中。在Web层,开发者可以使用 Tapestry、Wicket、Struts、WebWork、JSF、Spring MVC,或者其他运行在Web容器中的Web框架。

至于Web层的具体实现并不是本文的重点。本文将关注如何管理数据库连接,这也是Web应用中处理线程安全问题是经常要考虑的资源。数据库连接对象,比如连接、结果集、Statement、Hibernate Session,是有状态对象。当然,它们不是线程安全的,因此不能够同时供多个线程访问。在本文前面已经提到,开发者应尽量避免使用同步。无论是 synchronized关键字,还是那些同步类(Hashtable或Vector),应尽量避免使用。因此,如果使用可重入,则不用处理阻塞或死锁。

当然,通过可重入实现线程安全以访问数据库并不是件简单的工作。比如,有些开发者可能会在Servlet容器配置中添加过滤器。因此,在客户请求到来时,过滤器将创建JDBC连接或Hibernate Session,并借助于ThreadLocal类将它们绑定到当前线程中,从而供业务逻辑使用。如果直接使用J2EE API,则开发者除了需要做很多同业务逻辑无关的操作外,还需要管理事务、DB错误等等开发内容。请注意,这些同业务逻辑无关的操作的维护工作往往很费时间。

Spring的闯入

一些Java开发者可能听说过Spring提供的DAO抽象。当然,一些开发者也有可能使用过它。借助于Spring提供的模板,开发者能够使用DAO代码的重用。借助于Spring AOP,开发者还能够使用声明式事务。因此,本文来研究Spring是如何实现以线程安全方式访问RDBMS的。比如,Spring允许以JDBC、 Hibernate、JDO、iBATIS、TopLink等方式访问数据库。如下给出的实例是企业应用中很常见的情景。

首先,定义数据源和用于Hibernate SessionFactory。



<bean    >   <property name="locations">   <list>   <value>WEB-INF/jdbc.properties</value>   </list>   </property>   </bean>   <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">   <property name="driverClassName"> <value>${jdbc.driverClassName}</value> </property>   <property name="url"><value>${jdbc.url}</value></property>   <property name="username"><value>${jdbc.username}</value></property>   <property name="password"><value>${jdbc.password}</value></property>   </bean>   <bean id="sessionFactory" class=" org.springframework.orm.hibernate.LocalSessionFactoryBean">   <property name="dataSource">   <ref bean="dataSource"/>   </property>   <property name="mappingDirectoryLocations">   <list>   <value>classpath:</value>   </list>   </property>   <property name="hibernateProperties">   <props>   <prop key="hibernate.dialect">net.sf.hibernate.dialect.HSQLDialect</prop>   <prop key="hibernate.show_sql">true</prop>   </props>   </property>   </bean>

这是使用Hibernate的典型配置,即通过定义的数据源连接到数据库、通过本地 SessionFactory创建Hibernate SessionFactory。接下来,需要定义业务对象(实现对DB的访问)和事务管理器(通过Hibernate Session管理本地事务)。其中,业务对象暴露的方法能够在数据库中添加新的纪录,而事务管理器能够将方法包裹在事务中。它们的定义如下。



 public interface CustomerDAO {   public void createCustomer(Customer customer);   }   public class HibernateCustomerDAO implements CustomerDAO {   private HibernateTemplate hibernateTemplate = null;   public void setSessionFactory(SessionFactory sessionFactory) {   this.hibernateTemplate = new HibernateTemplate(sessionFactory, false);   }   public void createCustomer(Customer customer) {   this.hibernateTemplate.save(customer);   }   }

开发者应该已经看到,上述类使用了Spring提供的 HibernateTemplate。注意,模板的开发遵循了业界最佳实践,并且将一些同业务不相关,但J2EE API规定要处理的那些代码处理掉了。与此同时,它通过DAO抽象将受查异常转换为非受查异常。当然,Spring不只是为使用Hibernate提供模板,它还为JDBC、iBATIS、SqlMap、JDO、TopLink提供类似模板。由于这些模板类及其实例变量实现了可重入,即都是线程安全的,因此允许并发线程同时使用模板。使用这些模板不仅能够实现代码的重用,还提供了最佳实践。除了以线程安全方式访问DB外,模板还提供了其他很多有意义的内容。好了,来看看如何定义业务对象和事务管理器吧!



<bean >   <property name="sessionFactory"> <ref bean="sessionFactory"/> </property>   </bean>   <bean id="transactionManager" class=" org.springframework.orm.hibernate.HibernateTransactionManager">   <property name="sessionFactory"> <ref bean="sessionFactory"/></property>   </bean>   <bean id="customerDAO" class=" org.springframework.transaction.interceptor.TransactionProxyFactoryBean">   <property name="transactionManager"> <ref bean="transactionManager"/> </property>   <property name="target"><ref bean="customerDAOTarget"/></property>   <property name="transactionAttributes">   <props>    <prop key="create*">PROPAGATION_REQUIRED</prop>    <prop key="*">PROPAGATION_REQUIRED</prop>   </props>   </property>   </bean>

如果开发者对Spring中事务管理的配置不熟悉,则本文正好满足你们。首先,上述 Spring配置片断定义了业务对象HibernateCustomerDAO,它包裹了Hibernate SessionFactory。注意,默认时,Spring中定义的JavaBean都是单例的,HibernateCustomerDAO也不例外。这意味:多个线程可能同时执行createCustomer()方法。

其次,配置了Hibernate事务管理器,它包裹了同一Hibernate SessionFactory实例。在事务管理器每次执行时,它都会完成如下几件事情。其一,检查Hibernate Session是否绑定到当前线程。如果已绑定,则直接使用它。如果还未绑定,事务管理器将告知Hibernate SessionFactory创建新的Session,然后将创建的Session绑定到当前线程。其二,如果当前没有处于活动的事务,则事务管理器将启动新的事务,并将Session包裹进来。否则,直接参与到活动事务中。

整个过程是通过使用Spring提供的TransactionProxyFactoryBean实现的。当然,这是一种以声明方式实现的事务管理过程。 TransactionProxyFactoryBean能够为业务对象创建代理对象,从而通过事务管理器管理事务。当每次通过代理对象调用 createCustomer()方法时,事务管理器将根据事务属性管理事务。当前,Spring除了提供 HibernateTransactionManager事务管理器外,还为JDBC数据源、JDO、TopLink提供了相应的事务管理器。

再来看看业务对象吧!当调用createCustomer()方法时,HibernateTemplate将查找绑定到当前线程的Hibernate Session。由于上述配置文件片断传入到HibernateTemplate构建器的第二个参数为false,因此如果没有绑定Hibernate Session,则将抛出未受查异常。这对于那些未正确配置事务管理功能的场和特别有用(注意,事务管理器很重要)。一旦事务管理配置好后,Hibernate Session将绑定到当前线程,从而启动事务。请注意,HibernateTemplate不会去检查事务是否激活,也不会显示地启动或终止事务。也请注意,如果在声明的方法(事务属性中给出的)中抛出了未受查异常,则当前活动事务将回滚。
分享到:
评论

相关推荐

    多线程的批量线程同步解决方案

    "多线程的批量线程同步解决方案"这个标题暗示我们探讨的是如何在多线程环境下有效地管理和同步多个任务,确保数据一致性与程序正确性。下面将详细阐述相关知识点。 一、多线程基础 多线程是指在一个进程中同时执行...

    java线程同步实例

    根据提供的信息,我们可以深入探讨Java线程同步以及代码示例中的关键知识点。 ### Java线程同步实例:哲学家就餐问题 #### 1. 哲学家就餐问题简介 哲学家就餐问题是多线程编程中一个经典的同步问题。该问题描述为...

    Java多线程并发访问解决方案

    本文将深入探讨Java中的多线程并发访问解决方案,主要围绕以下几个核心知识点进行阐述: 1. **线程同步机制**: - **synchronized关键字**:Java中的synchronized提供了一种内置锁机制,它可以保证同一时间只有一...

    Java多线程同步机制在售票系统的实现

    Java的多线程同步机制为解决多线程环境下的数据一致性问题提供了有效的解决方案。在实际开发中,合理地运用这些机制能够显著提升程序的稳定性和安全性。尤其是在像售票系统这样的应用场景中,正确地使用同步机制可以...

    Java线程学习和总结

    本文档("Java线程学习和总结.htm")可能包含了更多关于线程的实例、源码分析和常见问题解决方案,你可以通过阅读来进一步加深对Java线程的理解。同时,"Java线程学习和总结.files"目录下的文件可能是与文章相关的...

    Java同步线程模型分析与改进

    本文针对Java同步线程模型存在的问题,提出了扩展`synchronized`关键字语法和支持超时检测的`wait()`方法的改进方案。这些改进不仅增强了Java多线程编程的灵活性,还提高了程序的性能、稳定性和可靠性。未来的研究...

    线程同步-mutex方案-Peterson方案

    本文将详细探讨线程同步中的两种经典解决方案:mutex方案和Peterson方案,以及它们在Java环境中的实现。 首先,我们来了解mutex(互斥锁)方案。Mutex是一种基本的同步原语,它的主要目标是防止多个线程同时进入...

    某电信项目多线程同步数据实例

    5. **并发工具类**:Java并发包(java.util.concurrent)中包含许多工具类,如Semaphore(信号量)、CountDownLatch(倒计时锁)、CyclicBarrier(回环栅栏)和Exchanger(交换器),它们是设计复杂多线程同步场景的...

    Java多线程-线程的安全问题与线程的同步机制介绍

    常见的线程同步机制有:同步块(Synchronized Block)、同步方法(Synchronized Method)、锁(Lock)、Atomic 变量等。 四、线程安全问题解决方案 (1)方式一:实现Runnable接口 可以通过实现Runnable接口来...

    Java线程入门

    ### Java线程入门知识点详解 #### 一、Java线程概览 - **目标读者**:本教程面向那些已经熟练掌握Java语言基本应用但对多线程编程尚不熟悉的程序员。 - **主要内容**:本教程将从零开始介绍Java线程的基础知识,...

    java多线程_java多线程下变量共享_

    Java多线程是Java编程中的重要概念,它允许...理解并掌握Java多线程下变量共享的原理和解决方案,有助于编写出高效、稳定的并发程序。在实际开发中,应结合具体业务场景选择合适的同步机制,以达到最佳性能和安全性。

    Java异步调用转同步方法实例详解

    Java异步调用转同步方法实例详解是指在Java中将异步调用转换为同步调用的技术,主要用于解决异步调用过程中的阻塞问题。异步调用是一种非阻塞的调用方式,调用方在调用过程中,不直接等待返回结果,而是执行其他任务...

    Java实现数据库迁移同步

    本文将详细讲解如何使用Java高效地实现数据库迁移同步。 首先,我们需要理解数据库迁移的基本概念。数据库迁移是指将数据从一个数据库系统迁移到另一个数据库系统的过程,或者在同一系统中不同版本之间进行数据转移...

    Java线程不安全分析,同步锁和Lock机制,哪个解决方案更好.docx

    ### Java线程不安全分析与解决方案探讨 #### 一、线程不安全的概念与原因 在多线程环境中,线程不安全是指当多个线程同时访问或修改同一份数据时,可能导致数据的一致性和准确性出现问题的情况。这种情况下,程序...

    java多线程导出excel(千万级别)优化

    Java多线程导出Excel是处理大数据量时的一种高效策略,尤其在面对千万...通过这种多线程和分批处理的方法,可以在不牺牲性能的前提下,安全地处理千万级别的数据导出,使得这种解决方案在生产环境中变得可行且稳定。

    java多线程代码demo

    Java的`java.util.concurrent`包提供了一系列高级并发工具,如Semaphore(信号量),CyclicBarrier(回环栅栏),CountDownLatch(计数器门锁),ExecutorService(线程池)等,它们为多线程编程提供了更灵活、高效...

    Java线程不安全分析,同步锁和Lock机制,哪个解决方案更好 (2).docx

    ### Java线程不安全分析与解决方案探讨 #### 一、线程不安全的概念与原因 在多线程环境中,线程不安全是指当多个线程同时访问或修改同一份数据时,可能导致数据的一致性和完整性受到破坏的现象。这种现象通常发生...

Global site tag (gtag.js) - Google Analytics