为了账号安全,请及时绑定邮箱和手机立即绑定

SSM实现高并发秒杀

标签:
Java

1、整体业务分析

1)展示商品所有列表;

图片描述

2)点击链接进入商品详情页;
图片描述

![图片描图片描述图片描述

2、DAO层设计与开发

2.1 Mysql&&NoSQL

​ 关系型数据库(Mysql/Oracle/SqlServer等)和非关系型数据库(HBase/redis/mongoDB/Neo4j等)。

​ 关系型和非关系型数据库的主要差异是数据存储的方式,关系型数据天然就是表格式的,非关系型数据是基于键值对的,通常存储在数据集中,就像文档、键值对或者图结构,所有非关系型数据存储天然就是分布式的,其真正的价值在操作的扩展性和大数据量处理。

​ SQL数据库支持对事务原子性细粒度控制,并且易于回滚事务,其在事务性操作上性能和稳定性比NoSQL好很多。并且目前事务机制是目前最可靠的落地方案。

2.2 Mysql实现秒杀业务的难点

​ 当多个用户同时参与到秒杀活动时,会出现“竞争”,对于Mysql来说难点在于"事务+行级锁"

start transaction

update 库存数量(出现竞争,行级锁:一条事务执行,其余事务全部在等待,直到commit)

insert 购买明细

Commit

该问题在高并发部分解决!

2.3 数据库设计

​ 将DDL(数据库定义语言)卸载了“/sql/sechema.sql”文件中,再对数据库进行操作。

​ 建立一个seckill数据库,2张表:seckill、success_killed

2.4 Mybatis实现DAO理论

​ Mybatis主要特点在于:参数 + SQL = Entity/List

​ Mybatis在哪些SQL:XML提供SQL

​ Mybaits如何实现接口: Mapper自动实现DAO接口, 通过Dao找Mapper文件下的同名的SQL语句,一个Dao对 应一个Mapper

2.5 Mybatis整合Spring理论

​ 整合目标:更少的编码(只写接口不写实现类);

​ 更少的配置(1、使用别名,通过org.seckill.entity包扫描将 org.seckill.entity.Seckill—>Seckill

​ 2、配置扫描,扫描sql配置文件:mapper需要的xml文件

​ 3、dao的实现,自动实现dao接口,自动注入spring容器);

​ 足够的灵活性(可以自己定义SQL语句、可以自由的通过表达式传递参数,最终实现结果集自动赋 值。)

2.6 DAO实体/接口设计以及基于Mybatis实现

​ (1)配置Mybatis全局属性"resources/mybatis-config.xml"。

​ (2)将数据库中的表——>Java中entity实体(org.seckill.entity),将表中的列与实体类中的属性一一对应

​ 如果业务逻辑中出现多对一的情况,需要在多方的实体类中定义一方的实体类。在秒杀系统中多个成功秒杀的实体对应的是同一个秒杀对象实体,所在在SuccessKilled类中定义Seckill实体类:private Seckill seckill。

​ (3)实体类对应的接口设计,常用的是实体下的增删改查操作。(org.seckill.dao)

​ (4)为DAO接口的方法提供SQL语句:“resources/mapper/*.xml”,namespace对应文件路径,id对应方法名,resultType对应返回值类型,parameterType对应参数类型,当出现多个参数的时候,接口方法中需要对参数进行注解,比如(@Param(“seckillId”) long seckillId, @Param(“killTime”)Date killTime)。

​ (5)Spring整合Mybatis,相关文件“resources/jdbc.properties”和“resources/spring/spring-dao”

​ 配置数据库相关参数“resources/jdbc.properties”、 数据库池连接、配置sqlSessionFactory对象、配置扫描Dao接口包,动态实现Dao接口,注入到spring容器中

3、Service层开发

3.1 秒杀Service接口设计

​ (1)业务接口要站在使用者的角度设置接口。

​ 从三个方面入手:方法定义的粒度(从用户操作的角度入手,一般每个button都会对应一个方法)、参数(越简练越好)、返回类型(return 类型/异常)。

(2)service需要的package

​ service包:用来存放service接口及其实现类"./service/Impl/SeckillServiceImpl";

​ exception包:存放service接口所需要的一些异常,比如重复秒杀、秒杀结束、内部错误等;

​ dto包:数据传输包 ,存放一个service和web之前的数据传递。

3.2 Spring IOC整合Service

​ Spring会通过对象工厂和依赖管理实现一个一致性的访问接口。

​ IOC的优点:对象创建统一托管、规范的生命周期管理、灵活的依赖注入、一致的获取对象(从容器中一致获取) 。

​ Spring IOC 的注入方式:XML(用bean来实现第三方类库,如DataSource, 需要命名空间配置,如context、aop、mvc等),注解(项目中自身开发使用的类,可以在代码中使用注解,如@Service @Controller等)、java配置类(自定义修改类库,不建议)。

​ 本项目中IOC使用:XML配置——>package-scan包扫描(扫描service包下所有使用注解的类型,这些类由Spring统一托管,不需要新建而是自动加载对象的实例到Spring 容器中)——>Annotation注解(默认使用注解来管理事务行为)

图片描述

3.3 声明式事务运用

​ 对事务的处理交由Spring进行统一的管理,当你的业务方法不发生异常(或者发生异常,但该异常也被配置信息允许提交事务)时,Spring就会让事务管理器提交事务,而发生异常(并且该异常不被你的配置信息所允许提交事务)时,则让事务管理器回滚事务。

​ 注意try…catch…中的异常,Spring 可能感知不到。

​ 采用注解的方式实现声明式事务@Transactional

使用注解控制事务方法的优点:
 1、开发团队达成一直约定,约定明确标注事务方法的编程风格。
 2、保证事务方法的执行时间尽可能短,不要穿插其他的网络操作(RPC/HTTP请求),或者剥离/封装到方法外面。
 3、不是所有的方法都需要事务,如只有一条修改操作、只读操作。

4、Web层开发

4.1 Restful接口设计与使用

​ 一种优雅的URI表达方式、能表示资源的状态(名词)和状态的转移。

​ GET用于查询操作,POST用于添加和修改操作,PUT用于修改操作,DELETE用于删除操作

​ URL设计:/模块/资源/{标志}/集合1/… 如:/user/{uid}/friends 好友列表

![image-20200402150122102](assets/image-202图片描述

4.2 Spring MVC 框架运作流程

4.2.1 Spring MVC理论

image-20200402150528468图片描述

返回一个页面

@RequestMapping(value = "/{seckillId}/detail", method = RequestMethod.GET)
public String detail(@PathVariable("seckillId") Long seckillId, Model model) {
    if (null == seckillId) {
        return "redirect:/seckill/list";
    }
    Seckill seckill = seckillService.getById(seckillId);
    if (null == seckill) {
        return "forward:/seckill/list";
    }
    model.addAttribute("seckill", seckill);//承载最终要返回给用户的数据,返回Seckill类数据
    return "detail";//view,在spring-web1里面配置

}

返回一个json(现在大部分开发使用的方法)

@RequestMapping(value = "/{seckillId}/exposer",
        method = {RequestMethod.POST,RequestMethod.GET},
        produces = {"application/json;charset=utf-8"})
@ResponseBody //最后数据会封装成一个json,返回json数据
public SeckillResult<Exposer> exposer (Long seckillId){
    SeckillResult<Exposer> result;
    try {

        Exposer exposer = seckillService.exportSeckillUrl(seckillId);
        result = new SeckillResult<Exposer>(true,exposer);

    } catch (Exception e) {
        logger.error(e.getMessage(),e);
        result = new SeckillResult<Exposer>(false,e.getMessage());
    }
    return result;
}

cookie访问

@RequestMapping(value = "/{seckillId}/{md5}/execution",
        method = {RequestMethod.POST,RequestMethod.GET},
        produces = {"application/json;charset=utf-8"})
@ResponseBody
public SeckillResult<SeckillExcution> excute(@PathVariable("seckillId") Long seckillId,
                                              @PathVariable("md5") String md5,
                                              @CookieValue(value = "killPhone" ,required = false) Long phone )
//killPhone会放入cookie中
{
  return null
}

4.2.2 Spring MVC的配置

(1)配置DispatcherServlet,文件在WEB-INF/web.xml

1、配置DispatcherServlet

2、配置springMVC需要加载的配置文件spring-*.xml, Mybatis->spring->springMVC

3、默认匹配所有的请求 /

(2)配置Spring MVC

1、开启springMVC注解模式,

1)、自动注册DefaultAnnotationHandlerMapping,AnnotationMethodHandlerAdapter
2)、提供一系列的功能:数字和日期的format @NumberFormat @DateTimeFormat ,XML和JSON的数据支持

2、静态资源默认servlet配置

1)、加入对静态资源的处理: js、gif、png
2)、允许使用"/"作为整体映射

3、配置jsp 显示ViewResolver

4、扫描web相关的bean

4.3 Controller开发

与前端进行交互,返回一个界面或者json给前端,现在公司多都会使用返回json给前端。

一般使用一个统一的数据返回格式,在本项目中采用org.seckill.dto.SeckillResult

4.4 前端设计

使用bootstrap+jquery这里不多赘述

包括jQuery的一些插件:jQuery cookie插件、jQuery countDown倒计时插件

5、高并发

5.1高并发点和高并发分析

(1)高并发主要发生在:查看详情页、获取系统时间(不需要优化)、查看暴露接口、执行秒杀操作。

​ (2)CDN是一个内容分发网络,用于加速用户获取数据,部署在离用户最近的节点上,不需要去访问互联网服务器,一般针对的是内容不变化的界面,导致获取秒杀接口不可以使用CDN,因为它会随着时间推移而改变,比如获取到接口执行秒杀等等,这种操作就适合用***后端的缓存redis缓存***。

 get from cache
 if null
   get db
   put cache
   locgoin

(3)image-20200402163316458

分析:
NoSQL , MQ等运维成本和稳定性不高;
开发成本:数据一致性、回滚方案等需要手动实现;
幂等性难保证:重复秒杀问题;
不适合新手的架构。

(4)为什么不用Mysql解决?

​ 当java控制事务进行事务行为时会有行级锁

​ 1)update减库存:网络延迟问题和GC操作
​ 2)insert购买明细:网络延迟问题和GC操作
​ 3)commit/rollback

优化方向:减少行级锁的持有时间

(5)优化总结

image-20200402164519350图片描述

5.2优化思路并实现

5.2.1redis后端缓存优化

​ redis实现秒杀接口暴露优化:

​ 1)在pom.xml 文件中添加依赖:redis客户端 Jedis,protostuff序列化依赖;

​ 2)创建RedisDao实现redis存取方法,具体实现:org.seckill.dao.cache.RedisDao

​ 3)优化在SeckillService中exportSeckillUrl方法:

//1、访问redis
Seckill seckill = redisDao.getSeckill(seckillId);
if (seckill == null){
    //2访问数据库
     seckill = seckillDao.queryById(seckillId);
    if (null==seckill){
        return new Exposer(false,seckillId);
    }else{
        //放入redis
        redisDao.putSeckill(seckill);
    }

}

5.2.2 秒杀操作优化

​ (1)如何判断update更新库存成功:

​ 条件:update自身没报错、客户端确认update影响行数

​ 优化思路:把客户端逻辑放在MySql 服务器端,避免网络延迟和GC

​ 两种解决方案:定制SQL方案,修改SQL源码,update /* + [auto commit] */

​ ***使用存储过程整个操作在Mysql端完成. resources/sql/seckill.sql

​ (2)mybatis+spring实现存储过程操作

​ 1)在SeckillDao接口中添加一个方法 void killByProcedure(Map<String,Object> paramMap);

​ 2) 在对应的Mapper SeckillDao.xml文件中添加;

<!--    使用mybatis调用存储过程-->
    <select id="killByProcedure" statementType="CALLABLE">
        call execute_seckill(
        #{seckillId,jdbcType=BIGINT,mode=IN},
        #{phone,jdbcType=BIGINT,mode=IN},
        #{killTime,jdbcType=TIMESTAMP,mode=IN},
        #{result,jdbcType=INTEGER,mode=OUT}
        );
    </select>

​ 3)实现service层和controller层编码

点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消