1.主类,for循环模拟1000个线程进行秒杀
import redis.clients.jedis.Jedis; import java.util.concurrent.LinkedBlockingDeque; import java.util.concurrent.ThreadPoolExecutor; import java.util.concurrent.TimeUnit; public class Application { //创建线程池 public static ThreadPoolExecutor pool= new ThreadPoolExecutor( 10,100,10, TimeUnit.SECONDS, new LinkedBlockingDeque<Runnable>() ); public static void main(String[] args) { //jedis操作redis,连接配置信息,本地连接 Jedis jedis = new Jedis("localhost", 6379); //登录redis密码 jedis.auth("abc123456"); //选择操作库 jedis.select(1); //设置秒杀商品和库存 jedis.set("kill_num", "50"); //删除上一次的秒杀用户信息,然后再次调用该方法时,记录新的秒杀用户信息ID。 jedis.del("kill_list"); //关闭连接 jedis.close(); //for循环,模拟秒杀环境,1000个线程任务 for (int i = 0; i < 1000; i++) { //线程任务加入到线程池里面 pool.execute(new KillTask()); } } }
2.秒杀任务类
import redis.clients.jedis.Jedis; import redis.clients.jedis.Transaction; public class KillTask implements Runnable{ @Override public void run() { Jedis jedis = new Jedis("localhost", 6379); jedis.auth("abc123456"); jedis.select(1); //watch监控key,获得数据的版本号 jedis.watch("kill_num", "kill_list"); int num = Integer.parseInt(jedis.get("kill_num")); //判断库存 if (num > 0) { //开启事务 Transaction transaction = jedis.multi(); transaction.decr("kill_num");//库存减一 transaction.rpush("kill_list", "9527");//记录用户ID,自己写的是9527 transaction.exec();//提交事务 } else { Application.pool.shutdown(); } jedis.close(); } }
3.运行主类时,需要开启redis->start.bat,然后运行成功后,打开redis客户端查看(需要连接),如下图:
因为使用redis的是db1数据库,所以查看db1数据库:
然后查看kill_num,此时键值是0,没有出现超买的现象:
最后查看kill_list,秒杀列表中出现秒杀商品的数量为50个,当初我们设置的也是50个商品数量。因为我们是写死的用户信息ID,所有都是9527这个用户:
好了,使用redis避免了商品秒杀的超卖问题!