十三、处理并发

1、事务的隔离级别

不考虑隔离级别出现的问题:

脏读:一个线程中的事务读到了另外一个线程中未提交的数据。

不可重复读:一个线程中的事务读到了另外一个线程中提交的update(更新)的数据。

虚读:一个线程中的事务读到了另外一个线程中提交的insert(插入)的数据。

事务的隔离级别:

1:READ UNCOMMITTED:脏读、不可重复读、虚读(幻读)都可能发生。

2:READ COMMITTED:避免脏读;不可重复读、虚读(幻读)都可能发生。

4:REPEATABLE READ:避免脏读、不可重复读,虚读(幻读)可能发生。

8:SERIALIZABLE:避免脏读、不可重复读、虚读(幻读)。

通过配置参数hibernate.connection.isolation=1|2|4|8进行设置。MySQL:默认4  Oracle:默认2

2、第二类更新丢失

 //线程1
@Test
public void test1(){
Session s = HibernateUtil.getSession();
Transaction tx = s.beginTransaction();
Customer c1 = s.get(Customer.class, 1);
c1.setName("泰斯特");
tx.commit();
s.close();
} //线程2
@Test
public void test2(){
Session s = HibernateUtil.getSession();
Transaction tx = s.beginTransaction();
Customer c1 = s.get(Customer.class, 1);
c1.setAge(20);
tx.commit();
s.close();
}

3、解决丢失更新的方案

3.1、悲观锁(不是锁,是一种处理态度。是hibernate的)

Hibernate认为丢失更新现象一定会发生。悲观锁用的是数据库的锁机制。(独占模式效率肯定不会高)

数据库的锁机制:(是真正的锁)

共享锁:可以有多把锁。

MySQL客户端线程1

MySQL客户端线程2

开启事务 strat transaction;

开启事务 strat transaction;

执行查询:select * from customes where id = 1 lock in share mode; 加共享锁

执行查询:select * from customes where id = 1 lock in share mode; 加共享锁

执行更新:update customers set name='张三' where id = 1;

此时会自动加排他锁。但是由于线程2没有释放共享锁,所以线程1处于等待状态。

commit;

当线程2释放共享锁,线程1就可以执行成功了。

排他锁(独占模式,也称为独占锁):只能有一把锁,在锁之前,不能有任何的锁。执行写操作(insert,update,delete)的时候会自动加锁。

MySQL客户端线程1

MySQL客户端线程2

开启事务 strat transaction;

开启事务 strat transaction;

执行查询:select * from customes where id = 1 for update; 加排他锁

执行查询:select * from customes where id = 1 lock in share mode ; 由于有线程1的排他锁存在,线程2将处于等待状态

执行更新:update customers set age = 50 where id = 1;

此时还没有释放排他锁

提交事务(释放排他锁) commit;

线程2显示查询结果,已经是线程1改过之后的数据了。

死锁:DEADLOCK

MySQL客户端线程1

MySQL客户端线程2

开启事务 strat transaction;

开启事务 strat transaction;

执行查询:select * from customes where id = 1 lock in share mode; 加共享锁

执行查询:select * from customes where id = 1 lock in share mode; 加共享锁

执行更新:update customers set name='张三' where id = 1;

此时会自动加排他锁。但是由于线程2没有释放共享锁,所以线程1处于等待状态。

执行更新:update customers set age=18 where id = 1;

此时会自动加排他锁。但是由于线程1没有释放共享锁,所以线程2处于等待状态。

结果:线程1等着线程2释放共享锁,而线程2等着线程1释放排他锁。就造成了死锁。

 //线程1
@Test
public void test1(){
Session s = HibernateUtil.getSession();
Transaction tx = s.beginTransaction();
Customer c1 = s.get(Customer.class, 1,LockMode.PESSIMISTIC_WRITE);//hibernate处理丢失更新的悲观锁解决方式
//Customer c1 = s.get(Customer.class, 1);
c1.setName("泰斯特");
tx.commit();
s.close();
} //线程2
@Test
public void test2(){
Session s = HibernateUtil.getSession();
Transaction tx = s.beginTransaction();
Customer c1 = s.get(Customer.class, 1,LockMode.PESSIMISTIC_WRITE);
//Customer c1 = s.get(Customer.class, 1);
c1.setAge(20);
tx.commit();
s.close();
}

3.2、乐观锁(不是锁,是一种处理态度)

Hibernate认为丢失更新现象不一定会发生。乐观锁使用的是版本号机制。

     private Integer id;
private String name;
private Integer age;
//================版本号==================
private Integer version;
//=======================================
//一对多关系映射:一个客户可以有多个订单
private Set<Order> orders = new HashSet<Order>(0); public Customer(){ }
      <class name="Customer" table="T_CUSTOMERS">
<id name="id" column="id">
<generator class="native"></generator>
</id>
<!-- 版本字段,用于解决丢失更新 -->
<version name="version"></version> <property name="name" column="NAME"></property>
<property name="age" column="AGE"></property>
<!-- 一对多关系映射:
set元素:
作用:映射集合元素
属性:
name:映射实体类中的集合属性
table:指定对应的表
key元素:它是set的子元素
作用:就是用于指定外键的
属性:
column:指定外键字段的名称
one-to-many元素:它是set的子元素
作用:描述当前实体映射文件和set中指定属性之间的关系。
属性:
class:指定是从表的实体类名称
-->
<set name="orders" table="T_ORDERS" >
<key column="CUSTOMER_ID"></key>
<one-to-many class="Order"/>
</set>
</class>

Java实战之02Hibernate-06处理并发的更多相关文章

  1. Java生鲜电商平台-高并发核心技术订单与库存实战

    Java生鲜电商平台-高并发核心技术订单与库存实战 一. 问题 一件商品只有100个库存,现在有1000或者更多的用户来购买,每个用户计划同时购买1个到几个不等商品. 如何保证库存在高并发的场景下是安 ...

  2. Java 面试知识点解析(二)——高并发编程篇

    前言: 在遨游了一番 Java Web 的世界之后,发现了自己的一些缺失,所以就着一篇深度好文:知名互联网公司校招 Java 开发岗面试知识点解析 ,来好好的对 Java 知识点进行复习和学习一番,大 ...

  3. Java多线程学习(七)并发编程中一些问题

    本节思维导图: 关注微信公众号:"Java面试通关手册" 回复"Java多线程"获取思维导图源文件和思维导图软件. 多线程就一定好吗?快吗?? 并发编程的目的就 ...

  4. JAVA NIO non-blocking模式实现高并发服务器

    JAVA NIO non-blocking模式实现高并发服务器 分类: JAVA NIO2014-04-14 11:12 1912人阅读 评论(0) 收藏 举报 目录(?)[+] Java自1.4以后 ...

  5. Java虚拟机JVM学习06 自定义类加载器 父委托机制和命名空间的再讨论

    Java虚拟机JVM学习06 自定义类加载器 父委托机制和命名空间的再讨论 创建用户自定义的类加载器 要创建用户自定义的类加载器,只需要扩展java.lang.ClassLoader类,然后覆盖它的f ...

  6. java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱

    java高级精讲之高并发抢红包~揭开Redis分布式集群与Lua神秘面纱 redis数据库 Redis企业集群高级应用精品教程[图灵学院] Redis权威指南 利用redis + lua解决抢红包高并 ...

  7. JAVA NIO non-blocking模式实现高并发服务器(转)

    原文链接:JAVA NIO non-blocking模式实现高并发服务器 Java自1.4以后,加入了新IO特性,NIO. 号称new IO. NIO带来了non-blocking特性. 这篇文章主要 ...

  8. 「小程序JAVA实战」springboot的后台搭建(31)

    转自:https://idig8.com/2018/08/29/xiaochengxujavashizhanspringbootdehoutaidajian31/ 根据下面的图,我们来建立下对应的sp ...

  9. 20145219 《Java程序设计》第06周学习总结

    20145219 <Java程序设计>第06周学习总结 教材学习内容总结 InputStream与OutputStream 串流设计 1.串流:Java将输入/输出抽象化为串流,数据有来源 ...

  10. Java多线程之同步集合和并发集合

    Java多线程之同步集合和并发集合 不管是同步集合还是并发集合他们都支持线程安全,他们之间主要的区别体现在性能和可扩展性,还有他们如何实现的线程安全. 同步集合类 Hashtable Vector 同 ...

随机推荐

  1. 第一个android程序所遇到问题

    1.工程package的命名空间与activity的命名空间不一致,导致setcontentview找不到layout文件 2.增加Button等控件后,Java.R中id必须在删除现有Java.R文 ...

  2. MSSQLSERVER数据库- 触发器

    参考了别人写的文章,我删除掉一些废话,只看一些我想看的信息.整理了一下,记录在这里,方便以后查阅! 1.当触发INSERT触发器时,新的数据行就会被插入到触发器表和inserted表中. 2.当触发d ...

  3. idea生成JAVADOC 报java.lang.IllegalArgumentException解决方案[终极]

    idea生成javadoc文档,总是会报  java.lang.IllegalArgumentException     at sun.net.www.ParseUtil.decode(ParseUt ...

  4. 基于华为Java编程规范的Eclipse checkStyle.xml

    发现项目组成员代码规范存在较大的问题,于是就在华为编程规范的基础上制定了这份checkStyle.xml文档,至于Eclipse怎么安装checkStyle插件以及该插件怎么使用请自行Google之. ...

  5. 【Android开发经验】使用Ant批量打包Android应用全然指南

    本文章由Socks完毕.博客地址:http://blog.csdn.net/zhaokaiqiang1992 转载请说明. 折腾了一下午.百度了一下午,最终实现了使用Ant对Android应用的批量打 ...

  6. Nginx+Tomcat动静态资源分离

    1 创建用户.用户组 useradd -g users www passwd www //设置密码,否则该用户不可用 groupadd -g 888 www //创建用户组 gpasswd -a ww ...

  7. 《Java并发编程实战》第十一章 性能与可伸缩性 读书笔记

    造成开销的操作包含: 1. 线程之间的协调(比如:锁.触发信号以及内存同步等) 2. 添加�的上下文切换 3. 线程的创建和销毁 4. 线程的调度 一.对性能的思考 1 性能与可伸缩性 执行速度涉及下 ...

  8. wcf自身作为宿主的一个小案例

    第一步:创建整个解决方案 service.interface:用于定义服务的契约(所有的类的接口)引用了wcf的核心程序集system.ServiceModel.dll service:用于定义服务类 ...

  9. Oracle database server 安装tips

    需要手动解压第二个包的文件合并到第一个包的相同目录中. 以12c为例,需要把 winx64_12102_SE2_database_1of2.zip和winx64_12102_SE2_database_ ...

  10. How to solve GM MDI cannot complete the installation

    Dear Joy, I have a problem using GM MDI diagnostic tool. When I installed it on my laptop, the tool ...