1:商品库存秒杀采用悲观锁Pessimistic-Lock主要好处是安全,充分利用了数据库的性能来做的一种锁机制。

悲观锁的实现:

(1)环境:mysql + jdbctemplate

(2)商品表goods:

DROP TABLE IF EXISTS `goods`;
CREATE TABLE `goods` (
`id` int(11) unsigned NOT NULL AUTO_INCREMENT,
`name` varchar(100) DEFAULT NULL COMMENT '商品名称',
`stock` int(11) unsigned NOT NULL COMMENT '商品库存',
`version` int(2) DEFAULT NULL COMMENT '版本号',
`token_time` datetime NOT NULL COMMENT '乐观锁时间戳',
PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=10 DEFAULT CHARSET=utf8; -- ----------------------------
-- Records of goods
-- ----------------------------
BEGIN;
INSERT INTO `goods` VALUES (1, 'product', 9999, 1, '2018-11-30 22:06:20');
COMMIT; SET FOREIGN_KEY_CHECKS = 1;

(3)DAO层代码:

package org.yugh.goodsstock.pessimistic_lock.repository;

import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Repository; import javax.annotation.Resource;
import java.util.List;
import java.util.Map; /**
* @author: YuGenHai
* @name: SeckRepository
* @creation: 2018/11/28 00:30
* @notes: 悲观锁DAO层
* @notes: 悲观锁需要注意mysql自带自动commit,用行锁需要开启事务 set transation 或者set autocommit =0
* 防止自动提交,set autocommit =1 自动提交
*/
@Repository
public class PessimisticLockRepository { /**
* 测试使用 {@link JdbcTemplate}
*/
@Resource
private JdbcTemplate jdbcTemplate; /**
* 获取现有库存量
* @param id
* @return
* @author yugenhai
*/
public int queryStock(long id) {
//开启事务
String lock = "set autocommit=0";
jdbcTemplate.update(lock);
//获得当前库存 并上锁
String sql = "select * from goods where id=1 for update";
List<Map<String,Object>> list = jdbcTemplate.queryForList(sql);
if(null != list && list.size() > 0){
Map<String,Object> map = list.get(0);
System.out.println("当前库存值: "+ map.get("stock"));
return Integer.valueOf(String.valueOf(map.get("stock")));
}
return 0;
} /**
* 还有库存量,并且要释放当前锁
* @author yugenhai
* @return
*/
public int updateStock() {
String update = "update goods set stock=stock-1 where id=1";
jdbcTemplate.update(update);
String unlock = "commit";
jdbcTemplate.update(unlock);
return 1;
} /**
* 商品被抢光后需要释放
* @author yugenhai
* @return
*/
public int unlock(){
String unlock = "commit";
jdbcTemplate.update(unlock);
return 1;
} }

(4)测试悲观锁:

package org.yugh.goodsstock.pessimistic_lock;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.yugh.goodsstock.pessimistic_lock.repository.PessimisticLockRepository; import javax.annotation.Resource; /**
* @author: YuGenHai
* @name: PessimisticLockTest
* @creation: 2018/11/28 00:32
* @notes: 悲观锁测试秒杀商品
*/
@RunWith(SpringRunner.class)
@SpringBootTest
public class PessimisticLockTest { @Resource
PessimisticLockRepository pessimisticLockRepository; /**
* STOCK库存总数,测试可以理解为购买者
* 表里的stock对应库存
*/
private static final int STOCK = 10000; /**
* 悲观锁秒杀商品
* @author yugenhai
*/
@Test
public void pessimisticLockTest() {
long beTime = System.currentTimeMillis();
for (int i = 0; i < STOCK; i++) {
//获得当前库存
//顺带上锁,开启事务
int stock = pessimisticLockRepository.queryStock(1);
if (stock > 0) {
//库存还有
//当前用户继续秒杀一个商品 并提交事务 释放锁
pessimisticLockRepository.updateStock();
System.out.println(new Thread().getName() + " 抢到了第 " + (i + 1) + " 商品");
} else {
//没有库存后释放锁
System.err.println(new Thread().getName() + " 抱歉,商品没有库存了!");
pessimisticLockRepository.unlock();
//break;
}
}
System.out.println("秒杀 "+ STOCK + " 件商品使用悲观锁需要花费时间:" + (System.currentTimeMillis() - beTime)); } }

(5)模拟10000个用户抢购9999个商品,最后一位用户没有抢到:

当前库存值: 8
Thread-9994 抢到了第 9992 商品
当前库存值: 7
Thread-9995 抢到了第 9993 商品
当前库存值: 6
Thread-9996 抢到了第 9994 商品
当前库存值: 5
Thread-9997 抢到了第 9995 商品
当前库存值: 4
Thread-9998 抢到了第 9996 商品
当前库存值: 3
Thread-9999 抢到了第 9997 商品
当前库存值: 2
Thread-10000 抢到了第 9998 商品
当前库存值: 1
Thread-10001 抢到了第 9999 商品
当前库存值: 0
秒杀 10000 件商品使用悲观锁需要花费时间:9922
Thread-10002 抱歉,商品没有库存了!
2018-12-01 00:51:06.914 INFO 9125 --- [ Thread-2] s.c.a.AnnotationConfigApplicationContext : Closing org.springframework.context.annotation.AnnotationConfigApplicationContext@f0da945: startup date [Sat Dec 01 00:50:56 CST 2018]; root of context hierarchy
2018-12-01 00:51:06.915 INFO 9125 --- [ Thread-2] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Shutdown initiated...
2018-12-01 00:51:06.920 INFO 9125 --- [ Thread-2] com.zaxxer.hikari.HikariDataSource : HikariPool-1 - Shutdown completed.

3:悲观锁和乐观锁项目

Java悲观锁Pessimistic-Lock常用实现场景的更多相关文章

  1. 乐观锁(optimistic locking)与悲观锁(pessimistic locking)

    首先,乐观锁(optimistic locking)与悲观锁(pessimistic locking)基本是针对数据处理来说,也就是跟数据库有关的术语,目的是为了解决并发处理时所遇到的相关性能问题,以 ...

  2. Java悲观锁和乐观锁

    悲观的并发策略——Synchronized互斥锁 互斥锁是最常见的同步手段,在并发过程中,当多条线程对同一个共享数据竞争时,它保证共享数据同一时刻只能被一条线程使用,其他线程只有等到锁释放后才能重新进 ...

  3. Java 悲观锁 synchronized (member){代码块}

    Java 如果遇到会出现高并发的情况,一般建议使用悲观锁 :synchronized (member){代码块}  需要对数据库进行修改或新增的时候,建议写上事务--@Transactional @T ...

  4. Java中的锁-悲观锁、乐观锁,公平锁、非公平锁,互斥锁、读写锁

    总览图 如果文中内容有错误,欢迎指出,谢谢. 悲观锁.乐观锁 悲观锁.乐观锁使用场景是针对数据库操作来说的,是一种锁机制. 悲观锁(Pessimistic Lock):顾名思义,就是很悲观,每次去拿数 ...

  5. Java多线程:乐观锁、悲观锁、自旋锁

    悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁.传统的关系型数据 ...

  6. Java乐观锁、悲观锁

    乐观锁 乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号 ...

  7. Hibernate、乐观锁和悲观锁

    悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁.传统的关系型数据 ...

  8. 乐观锁&悲观锁

    悲观&乐观,只是对数据加锁的时机与粒度. 悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这 ...

  9. Hibernate乐观锁、悲观锁和多态

     乐观锁和悲观锁 悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁 ...

随机推荐

  1. 可用的NTP服务器地址

    国内可用的Internet时间同步服务器地址(NTP时间服务器)好在阿里云提供了7个NTP时间服务器也就是Internet时间同步服务器地址 ntp1.aliyun.comntp2.aliyun.co ...

  2. 微服务-springboot-读写分离(多数据源切换)

    为什么需要读写分离 当项目越来越大和并发越来大的情况下,单个数据库服务器的压力肯定也是越来越大,最终演变成数据库成为性能的瓶颈,而且当数据越来越多时,查询也更加耗费时间,当然数据库数据过大时,可以采用 ...

  3. python中的 == 和 is 的区别

    == 比较的是两边的值 is 比较的是两边的内存地址  通过 id()获取内存地址 小数据池:我们使用过的值存储在小数据池中,供其他数据使用. 小数据池仅限于 数字 和 字符串: 数字的小数池范围  ...

  4. 用PHP抓取百度贴吧邮箱数据

    注:本程序可能非常适合那些做百度贴吧营销的朋友. 去逛百度贴吧的时候,经常会看到楼主分享一些资源,要求留下邮箱,楼主才给发. 对于一个热门的帖子,留下的邮箱数量是非常多的,楼主需要一个一个的去复制那些 ...

  5. S7-300CPU存储器介绍及存储卡使用

    1. S7 300存储区概述 S7-300 PLC的存储区可以划分为四个区域:装载存储器(Load Memory).工作存储器(Work Memory). 系统存储器(System Memory)和保 ...

  6. python3+pyQt5+QtDesignner实现窗口化猜数字游戏

    描述:使用QtDesignner设计界面,pyQt5+python3实现主体方法制作的猜数字游戏. 游戏规则:先选择游戏等级:初级.中级.高级.魔鬼级,选择完游戏等级后点击“确定”,然后后台会自动生成 ...

  7. C语言学习书籍推荐《C语言入门经典(第5版)》下载

    霍尔顿 (Ivor Horton) (作者), 杨浩 (译者) 下载地址:点我 C语言是每一位程序员都应该掌握的基础语言.C语言是微软.NET编程中使用的C#语言的基础:C语言是iPhone.iPad ...

  8. mysql in与exists问题剖析

    1 问题描述 ​ 发布当天发现一个日志分析的sql,在测试环境上执行良好,1秒内,而在线上环境上,执行要13秒左右. 嵌套sql一步一步分析后,发现出在in上,因时间紧迫,来补不及具体分析原因,尝试使 ...

  9. Fiolki题解

    问题 B: Fiolki 时间限制: 3 Sec  内存限制: 128 MB 题目描述 化学家吉丽想要配置一种神奇的药水来拯救世界. 吉丽有n种不同的液体物质,和n个药瓶(均从1到n编号).初始时,第 ...

  10. secureCRT背景颜色

    secureCRT是我们常用的linux远程登录软件,某些情况下我们安装的secureCRT工具可能默认背景颜色为白色,使用时很容易造成眼睛的疲劳,网上有些教程也只是修改当前登录窗口背景颜色,新的连接 ...