1.首先我们新建一个Controller用于秒杀:

package com.imooc.Controller;

import com.imooc.service.impl.SeckillServiceImpl;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; /**
* Created by zhongliahi on 2018/6/11.
* 秒杀测试
*/
@RestController
@RequestMapping(value = "/skill")
@Slf4j
public class SeckillController { @Autowired
private SeckillServiceImpl seckillService; //@PathVariable 可以将 URL 中占位符参数绑定到控制器处理方法的入参中
// URL 中的 {xxx} 占位符可以通过@PathVariable(“xxx“) 绑定到操作方法的入参中。
@GetMapping(value = "/query/{productId}")
public String query(@PathVariable String productId) throws Exception{
return seckillService.querySeckillProductInfo(productId);
} @GetMapping("/order/{productId}")
public String skill(@PathVariable String productId) throws Exception{
log.info("秒杀----productId:"+productId);
seckillService.orderProductMockDiffUser(productId); return seckillService.querySeckillProductInfo(productId);
}
}

  

2.建立一个Service

package com.imooc.service;

/**
* Created by zhongliahi on 2018/6/11.
*/
public interface SeckillService { String queryMap(String productId); String querySeckillProductInfo(String productId); void orderProductMockDiffUser(String productId);
}

3.实现Service

package com.imooc.service.impl;

import com.imooc.Exception.SellException;
import com.imooc.enums.ExceptionEnum;
import com.imooc.service.SeckillService;
import com.imooc.util.KeyUtils;
import org.springframework.stereotype.Service; import java.util.HashMap;
import java.util.Map; /**
* Created by zhonglihai on 2018/6/11.
* 秒杀Serviceimpl
* 演示
*/
@Service
public class SeckillServiceImpl implements SeckillService { /**
* 秒杀特价 1000000份
* @param productId
* @return
*/
static Map<String,Integer> products;
static Map<String,Integer> stock;
static Map<String,String> orders; static{
/**
* ,模拟多个表,商品信息表,库存表,秒杀成功订单表
*/
products =new HashMap<>();
stock=new HashMap<>();
orders=new HashMap<>();
products.put("123",1000000);
stock.put("123",1000000);
} @Override
public String queryMap(String productId) {
return "活动特价,限量:"+products.get(productId)+",还剩:"+stock.get(productId)
+"份"+",成功下单用户数:"+orders.size()+"人。";
} @Override
public String querySeckillProductInfo(String productId) {
return this.queryMap(productId);
} /**
* 主要秒杀的逻辑
* @param productId
*/
@Override
public synchronized void orderProductMockDiffUser(String productId) {
//查询该商品库存,为0则活动结束
int stockNum=stock.get(productId);
if(stockNum==0){
throw new SellException(ExceptionEnum.SECKILL_OVER);
}else{
//2.下单(模拟不同用户opendid不同)
orders.put(KeyUtils.getUniqueKey(),productId); //3.减库存
stockNum=stockNum-1;
try{
Thread.sleep(100);
} catch (InterruptedException e) {
e.printStackTrace();
}
stock.put(productId,stockNum);
}
}
}

关于压测:

  压力测试是一种基本的质量保证行为,在秒杀活动中更为重要,能有效测量超卖(卖出去的比库存的多)、少卖(有买了但是没卖)等现象,目前主流的压测工具有Jmeter、LoadRunner等,老一点的有apache ab,正好本人机器装有Apace服务,因此使用apache  ab做压测。

项目中,我们只添加了一件商品,productId为123,启动项目,在浏览器中查询,URL:http://127.0.0.1:8080/sell/skill/query/123,结果如下:

可以看到,项目能正常访问,查询的库存为1000000份。现在我们咋浏览器上进行秒杀,URL:http://127.0.0.1:8080/sell/skill/order/123

我们已经秒杀了一件商品,那么如何实现高并发秒杀呢?这就需要使用上面介绍的Apache ab进行压测

使用方法:安装Apache Http Services 后,配置相关环境变量,保证能在命令行直接调用。

测试命令:ab -n 1000 -c 10 http://127.0.0.1:8080/sell/skill/order/123

  其中ab表示在命令行调取apache ab压测工具,-n表示发起1000条请求,-c 表示10个并发

  http://127.0.0.1:8080/sell/skill/order/123 :表示测试的URL.

  (注意:压测会占用大量电脑资源,特别是并发大的时候)

结果:

可以发现,秒杀还是比较快的,仅用了19秒。现在我们来查看库存;

-----------------------------------------

重点:查看库存发现虽然秒杀都成功了,但是库存量与下单成功量之和与总量不对应:999878+1000>1000000,出现了超卖现象。

下面我们在秒杀方法上加上synchronized关键字,修SeckillServiceImpl,对秒杀方法上锁

然后重启项目,在此重新查询库存:

没问题!

继续并发秒杀:

测试完成,明显可以感觉到,访问慢了很多,用了100多秒,因为我们使用了资源锁,保证每次只有一个线程去调用它。

现在我们在来查看库存。

重点:可以发现,库存与下单都是正确的,使用synchronized是一种资源控制的解决办法

那么,秒杀中直接使用synchronized进行锁控制有什么不好的地方呢?

  1.无法做到细粒度的控制,在测试中,我们只有一个商品,如果有多个商品呢?

  多个商品参与秒杀活动,有的人秒杀商品A、有的秒杀商品B,都要走秒杀方法,使用synchronized

  会一样的慢。

  2.只支持单点(单机、服务器环境),无法做到水平扩展,如果项目使用负载均衡,会出现混乱。

那么,又有什么好的办法可以解决上面提到的问题?

答案当然是有,那就是分布式锁。

  

  

synchronized 控制并发(活动秒杀)的更多相关文章

  1. 【总结】瞬时高并发(秒杀/活动)Redis方案(转)

    转载地址:http://bradyzhu.iteye.com/blog/2270698 1,Redis 丰富的数据结构(Data Structures) 字符串(String) Redis字符串能包含 ...

  2. 【总结】瞬时高并发(秒杀/活动)Redis方案

    1,Redis 丰富的数据结构(Data Structures) 字符串(String) Redis字符串能包含任意类型的数据 一个字符串类型的值最多能存储512M字节的内容 利用INCR命令簇(IN ...

  3. java并发编程学习:用 Semaphore (信号量)控制并发资源

    并发编程这方面以前关注得比较少,恶补一下,推荐一个好的网站:并发编程网 - ifeve.com,上面全是各种大牛原创或编译的并发编程文章. 今天先来学习Semaphore(信号量),字面上看,根本不知 ...

  4. Nodejs - 如何用 eventproxy 模块控制并发

    本文目标 本文的目标是获取 ZOJ 1001-1010 每道题 best solution 的作者 id,取得数据后一次性输出在控制台. 前文 如何用 Nodejs 分析一个简单页面 我们讲了如何用 ...

  5. Java--Semaphore控制并发线程数量

    package com; import java.util.concurrent.Semaphore; /** * Created by yangyu on 16/11/28. */ /** * Se ...

  6. entity framework如何控制并发

     entity framework如何控制并发 针对字段http://msdn.microsoft.com/en-us/library/vstudio/bb738618(v=vs.100).aspx ...

  7. async和enterproxy控制并发数量

    聊聊并发与并行 并发我们经常提及之,不管是web server,app并发无处不在,操作系统中,指一个时间段中几个程序处于已经启动运行到完毕之间,且这几个程序都是在同一处理机上运行,并且任一个时间点只 ...

  8. [Go] golang无缓冲通道实现工作池控制并发

    展示如何使用无缓冲的通道创建一个goroutine池,控制并发频率1.无缓冲通道保证了两个goroutine之间的数据交换2.当所有的goroutine都忙的时候,能够及时通过通道告知调用者3.无缓冲 ...

  9. nodejs高并发大流量的设计实现,控制并发的三种方法

    nodejs高并发大流量的设计实现,控制并发的三种方法eventproxy.async.mapLimit.async.queue控制并发Node.js是建立在Google V8 JavaScript引 ...

随机推荐

  1. CentOS 7运维管理笔记(8)----Apache基于域名的虚拟主机配置

    使用基于域名的虚拟主机配置是比较流行的方式,可以在同一个IP上配置多个域名并且都通过80端口访问. (1) 在网卡 eth0的第五个接口上配置 192.168.1.215 这个地址: (2) 配置/e ...

  2. Android 触发Button按钮事件的三种方式

    1.新创建一个类 2.使用内部类 3.当多个button按钮时,为简化代码而创建的实例listener 贴代码: MainActivity.Java  文件: package com.android. ...

  3. 3 TFRecord样例程序实战

    将图片数据写入Record文件 # 定义函数转化变量类型. def _int64_feature(value): return tf.train.Feature(int64_list=tf.train ...

  4. 千里之堤毁于蚁穴(慎用HD Wallets)

    转自:http://blog.sina.com.cn/s/blog_12ce70a430102vbu9.html 千里之堤毁于蚁穴(慎用HD Wallets) -- 随机系列谈之四 现在我们都该明白, ...

  5. python 案例一(电话铺)

    经过自己努力,做了一个简单的电话铺的录入和查询小程序,比较简单,喜欢的朋友可以练练手. 题目: 创建你自己的命令行 地址簿 程序.在这个程序中,你可以添加.修改.删除和搜索你的联系人(朋友.家人和同事 ...

  6. 关于windows server 里Let's Encrypt续订的问题

    引言 Let's Encrypt是什么就不详细说了,它是免费的https证书,优点就是免费,缺点就是每三个月就要自己续上.今天主要介绍的是续上有效期的环节. 1.安装certify 下载地址: htt ...

  7. PPTP has become obsolete

    https://www.ovpn.com/en/blog/pptp-has-become-obsolete/ PPTP has become obsolete What is PPTP? PPTP s ...

  8. PPTP vs. OpenVPN vs. L2TP/IPsec vs. SSTP

    Which is the Best VPN Protocol? PPTP vs. OpenVPN vs. L2TP/IPsec vs. SSTP Want to use a VPN? If you’r ...

  9. Python初学者第十一天 文件处理_batch

    11day 文件的操作分为读.写.修改 1.读: f = open(file='D:\新建文本文档.txt',mode='r',encoding='gbk') data = f.read() prin ...

  10. 第六周 day6 python学习笔记

    1.Python面向对象编程OOP(Object Oriented Programming) 封装:可以隐藏实现细节,使代码模块化 继承:可以扩展已存在的代码模块,可以使代码实现重用 多态:一种接口, ...