• Hibernate的悲观锁和乐观锁


             前一篇博客我们从数据库角度分析,锁可以分为三种,分别为共享锁,独占锁和更新锁。我们从程序的角度来看锁可以分为两种类型,悲观锁和乐观锁,Hibernate提供对这两种锁 的支持,我们来了解一下Hibernate如何实现这两种锁。


    一、悲观锁 Pessimistic Locking

        

          通常由数据库机制实现,在查询的整个过程中把数据锁住,只要事务不释放(提交/回滚),那么其他任何用户都不能查看或修改数据,这种锁的方式是比较简单、直接。从开始就讲数据全部锁上,这种锁主要针对并发修改造成数据不一致的问题,但同时也会造成死锁的发生。

         适用场景:适合短事务

         示例:两个用户同时去到同一个数据100,用户1先将数据减20,这时数据库里的是80,用户2刚才读取到的是100,现在用户2,这样就会造成数据混乱。采用锁的方式解决这种问题。

     悲观锁:在用户1读取数据的时候,用锁将数据锁上,而用户2读取不到数据,只要用户1将数据修改后并提交,才释放锁,此时用户2才能读取到数据,而这时候读取到的是用户1修改后的数据,也就解决了数据混乱的问题了。

    实体类:

    public class Inventory {
    	private String itemNo;
    	private String itemName;
    	private int quantity;
    	
    	
    	public void setItemNo(String itemNo) {
    		this.itemNo = itemNo;
    	}
    	public String getItemName() {
    		return itemName;
    	}
    	public void setItemName(String itemName) {
    		this.itemName = itemName;
    	}
    	public int getQuantity() {
    		return quantity;
    	}
    	public void setQuantity(int quantity) {
    		this.quantity = quantity;
    	}
    	
    	
    	
    }


    配置文件:

    <hibernate-mapping>
    	<class name="com.bjpowernode.hibernate.Inventory" table="t_inventory">
    		<id name="itemNo">
    			<generator class="assigned"/>
    		</id>
    		<property name="itemName"/>
    		<property name="quantity"/>
    		
    	</class>
    	
    </hibernate-mapping>


    LockModel(使用UPGRADE)

    public void testLoad1(){
    		Session session = null;
    		try{
    			session=HibernateUtils.getSession();
    			session.beginTransaction();
    			
    			Inventory inv =(Inventory)session.load(Inventory.class, "1001",LockMode.UPGRADE);
    			System.out.println("opt1----ItemNO ="+inv.getItemNo());
    			System.out.println("opt1----ItemNname ="+inv.getItemName());
    			System.out.println("opt1----Quantity ="+inv.getQuantity());
    			
    			session.beginTransaction().commit();
    		}catch(Exception e){
    			e.printStackTrace();
    			session.getTransaction().rollback();
    		}finally{
    			HibernateUtils.closeSession(session);
    		}
    	}


    二、乐观锁OptimisticLocking

         

            不是锁,是一种冲突检测机制,乐观锁的并发性要好于悲观锁。常用的方式可以使用数据版本的方式(version)实现,一般是在数据库中加入一个version字段,在读取数据的时候将version字段读取出来,在保存数据的时候判断version的值是否小于数据库中version的版本,如果小于不予更新,否则更新数据。

    实体类:

         在实体类配置中添加上版本的映射

    public class Inventory {
    	private String itemNo;
    	private String itemName;
    	private int quantity;
    	private int version;
    	
    	public int getVersion() {
    		return version;
    	}
    	public void setVersion(int version) {
    		this.version = version;
    	}
    	public String getItemNo() {
    		return itemNo;
    	}
    	public void setItemNo(String itemNo) {
    		this.itemNo = itemNo;
    	}
    	public String getItemName() {
    		return itemName;
    	}
    	public void setItemName(String itemName) {
    		this.itemName = itemName;
    	}
    	public int getQuantity() {
    		return quantity;
    	}
    	public void setQuantity(int quantity) {
    		this.quantity = quantity;
    	}	
    	
    }


    配置文件:

    <hibernate-mapping>
        <class name="com.bjpowernode.hibernate.Inventory" table="t_inventory" optimistic-lock="version">
    		<id name="itemNo">
    			<generator class="assigned"/>
    		</id>
    		<version name="version"/>
    		<property name="itemName"/>
    		<property name="quantity"/>
    		
        </class>	
    </hibernate-mapping>


      使用方式还是和之前的一样,这样在更新之前就会跟数据库中版本对比,能够更好的解决数据混乱的问题。


    总结:

          数据库添加锁的两种方式,悲观锁简单明了,它将一切都以悲观的眼光来看待,认为一切都是并发的,而且当数据库很大或者遇到问题就很容易造成死锁。乐观锁的方式更加和谐,能够更好的处理并发问题。



  • 相关阅读:
    尬聊攻略丨过年回家,你最怕被亲戚问什么?
    有人被盗刷900次? 支付宝发布年终“神反转”盘点
    470余万条疑似12306用户数据遭贩卖 嫌疑人被刑拘
    利用Python实现对Web服务器的目录探测
    栈空间溢出
    KB奇遇记(6):搞笑的ERP项目团队
    探寻不同版本号的SDK对iOS程序的影响
    阅读《Android 从入门到精通》(24)——切换图片
    leetcode
    Grunt的配置及使用(压缩合并js/css)
  • 原文地址:https://www.cnblogs.com/zsswpb/p/6329432.html
Copyright © 2020-2023  润新知