分享一个整合 SSM 框架的高并发和商品秒杀项目

项目开发流程项目环境的搭建项目效果图项目的运行《Netty 实现原理与源码解析 —— 精品合集》《Spring 实现原理与源码解析 —— 精品合集》《MyBatis 实现原理与源码解析 —— 精品合集》《Spring MVC 实现原理与源码解析 —— 精品合集》《Spring Boot 实现原理与源码解析 —— 精品合集》《数据库实体设计合集》《Java 面试题 —— 精品合集》《Java 学习指南 —— 精品合集》
一个整合SSM框架的高并发和商品秒杀项目,学习目前较流行的Java框架组合实现高并发秒杀API
项目开发流程
本项目很适合学习一些技术的基础,这个项目的开发分为几个流程,很基础地教你接触到一个相对有技术含量的项目Java高并发秒杀API之业务分析与DAO层Java高并发秒杀API之web层Java高并发秒杀API之Service层Java高并发秒杀API之高并发优化按照上面几个流程走下去,你要有基本的Maven认识以及Java语法的一些概念,要不然可能不太理解
其实这几个流程也就是开发的流程,首先从DAO层开始开发,从后往前开发,开始Coding吧!
项目总结可能比较长,由于公众号文章的字数限制今天只能先讲解第一节,如果这篇文章看得人多并且点赞数较高的话明天就继续更新,哈哈~
项目环境的搭建
操作系统 : Ubuntu 17.04IDE :IntelliJ IDEA 2016.2.5 x64 用Eclipse也一样的,工具时靠人用的JDK : JDK1.8 建议使用JDK1.7以上版本,有许多语法糖用着挺舒服的Web容器 : Tomcat 8.0数据库 :Mysql-5.6.17-WinX64 实验性的项目用Mysql就足够啦依赖管理工具 : Maven 管理jar包真的很方便这里列出的环境不是必须的,你喜欢用什么就用什么,这里只是给出参考,不过不同的版本可能会引起各种不同的问题就需要我们自己去发现以及排查,在这里使用Maven的话时方便我们管理JAR包,我们不用跑去各种开源框架的官网去下载一个又一个的JAR包,配置好了Maven后添加pom文件坐标就会从中央仓库下载JAR包,如果哪天替换版本也很方便项目效果图
秒杀商品列表

秒杀结束提示界面

开始秒杀提示界面

重复秒杀提示界面

秒杀成功提示界面

项目的运行
下载
Download Zip`或者 `git clone
git clone https://github.com/Sunybyjava/seckill.git

导入到IDE
这里因为是使用IDEA创建的项目,所以使用IDEA直接打开是很方便的,提前是你要配置好maven的相关配置,以及项目JDK版本,JDK版本必须在1.8以上,因为在项目中使用了Java8的LocalDateTime以及LocalDate,所以低于这个版本编译会失败的
IDEA直接在主界面选择Open,然后找到项目所在路径,点击pom.xml打开就可以了Eclipse 这个项目是基于IDEA创建,我这里把项目转成了Eclipse的项目,如果你使用Eclipse的话也可以直接导入,只是步骤更繁琐一点,Eclipse导入步骤(一)Java高并发秒杀APi之业务分析与DAO层代码编写
构建项目的基本骨架首先我们要搭建出一个符合Maven约定的目录来,这里大致有两种方式,第一种:第一种使用命令行手动构建一个maven结构的目录,当然我基本不会这样构建mvn archetype:generate -DgroupId=com.suny.seckill -DartifactId=seckill -Dpackage=com.suny.seckill -Dversion=1.0-SNAPSHOT -DarchetypeArtifactId=maven-archetype-webapp

这里要注意的是使用archetype:generate进行创建,在Maven老版本中是使用archetype:create,现在这种方法已经被弃用了,所以使用命令行创建的话注意了,稍微解释下这段语句的意思,就是构建一个一个maven-archetype-webapp骨架的Webapp项目,然后groupId为com.suny.seckill,artifactId为seckill,这里是Maven相关知识,可以按照自己的情况进行修改
2.第二种直接在IDE中进行创建,这里以IDEA为例点击左上角File>New>Project>Maven然后在里面勾选Create from archetype,然后再往下拉找到org.apache.cocoon:cocoon-22-archetype-webapp,选中它,注意要先勾选那个选项,否则选择不了,然后点击Next继续
+然后就填写你的Maven的那几个重要的坐标了,自己看着填吧

+再就配置你的Maven的相关信息,默认应该是配置好的

+之后就是点Finsh,到此不出意外的话就应该创建成功了
构建pom文件
项目基本的骨架我们就创建出来了,接下来我们要添加一些基本的JAR包的依赖,也就是在pom.xml中添加各种开源组件的三坐标了

4.0.0
com.suny.seckill
seckill
1.0-SNAPSHOT
seckill Maven Webapp
http://maven.apache.org

junit
junit
4.12
test

ch.qos.logback
logback-classic
1.1.7

org.slf4j
slf4j-api
1.7.21

org.apache.logging.log4j
log4j-core
2.6.1


mysql
mysql-connector-java
5.1.39

c3p0
c3p0
0.9.1.2

org.mybatis
mybatis
3.4.2

org.mybatis
mybatis-spring
1.3.1

taglibs
standard
1.1.2

jstl
jstl
1.2

com.fasterxml.jackson.core
jackson-databind
2.8.5

javax.servlet
javax.servlet-api
3.1.0

org.springframework
spring-core
4.3.6.RELEASE

org.springframework
spring-beans
4.3.6.RELEASE

org.springframework
spring-context
4.3.6.RELEASE

org.springframework
spring-jdbc
4.3.7.RELEASE

org.springframework
spring-tx
4.3.6.RELEASE

org.springframework
spring-web
4.3.6.RELEASE

org.springframework
spring-webmvc
4.3.7.RELEASE

org.springframework
spring-test
4.2.7.RELEASE

seckill

建立数据库
在根目录下有一个sql文件夹里面有一个sql数据库脚本,如果你不想自己手写的话就直接导入到你的数据库里面去吧,不过还是建议自己手写一遍加深印象
— 整个项目的数据库脚本
— 开始创建一个数据库
CREATE DATABASE seckill;
— 使用数据库
USE seckill;
— 创建秒杀库存表
CREATE TABLE seckill(
`seckill_id` BIGINT NOT NULL AUTO_INCREMENT COMMENT 商品库存ID,
`name` VARCHAR(120) NOT NULL COMMENT 商品名称,
`number` INT NOT NULL COMMENT 库存数量,
`start_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP() COMMENT 秒杀开启的时间,
`end_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP() COMMENT 秒杀结束的时间,
`create_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP() COMMENT 创建的时间,
PRIMARY KEY (seckill_id),
KEY idx_start_time(start_time),
KEY idx_end_time(end_time),
KEY idx_create_time(create_time)
)ENGINE =InnoDB AUTO_INCREMENT=1000 DEFAULT CHARSET=utf8 COMMENT=秒杀库存表;

— 插入初始化数据

insert into
seckill(name,number,start_time,end_time)
values
(1000元秒杀iphone6,100,2016-5-22 00:00:00,2016-5-23 00:00:00),
(500元秒杀iPad2,200,2016-5-22 00:00:00,2016-5-23 00:00:00),
(300元秒杀小米4,300,2016-5-22 00:00:00,2016-5-23 00:00:00),
(200元秒杀红米note,400,2016-5-22 00:00:00,2016-5-23 00:00:00);

— 秒杀成功明细表
— 用户登录相关信息
create table success_killed(
`seckill_id` BIGINT NOT NULL COMMENT 秒杀商品ID,
`user_phone` BIGINT NOT NULL COMMENT 用户手机号,
`state` TINYINT NOT NULL DEFAULT -1 COMMENT 状态标示:-1无效 0成功 1已付款,
`create_time` TIMESTAMP NOT NULL COMMENT 创建时间,
PRIMARY KEY (seckill_id,user_phone), /*联合主键*/
KEY idx_create_time(create_time)
)ENGINE =InnDB DEFAULT CHARSET =utf8 COMMENT =秒杀成功明细表

在建立数据库的,如果按照我这里的数据库脚本建立的话应该是没问题的,但是我按照视频里面的数据库脚本建表的话发生了一个错误

这个报错看起来比较的诡异,我仔细检查sql也没有错误,它总提示我end_time要有一个默认的值,可我记得我以前就不会这样,然后视频里面也没有执行错误,然后我感觉可能时MySQL版本的差异,我查看了下我数据库版本,在登录Mysql控制台后输入指令,在控制台的我暂时知道的有两种方式:
select version();
select @@version;

我的输出结果如下:

进控制台就已经可以看到版本了,我的Mysql是5.7的,以前我用的时5.6的,然后去Google上搜索了下,找到了几个答案,参考链接:https://stackoverflow.com/questions/9192027/invalid-default-value-for-create-date-timestamp-fieldmysql官方的解释:https://dev.mysql.com/doc/refman/5.7/en/sql-mode.html#sqlmode_no_zero_datehttps://stackoverflow.com/questions/34570611/mysql-community-5-7-invalid-default-value-datetime-field-type
总结出来一句话就是:mysql 5.7中,默认使用的是严格模式,这里的日期必须要有时间,所以一定要给出默认值,要么就修改数据库设置然后网友评论里总结出来的几种解决办法,未经测试!:下次有问题一定要先看一下评论!!!create不了的同学,可以这样写:`start_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 秒杀开始时间,
`end_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 秒杀结束时间,
`create_time` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT 创建时间,
关于timestamp的问题,需要先运行 set explicit_defaults_for_timestamp = 1,否则会报invalid default value错误还需要注意的是SQL版本的问题会导致视频中seckill表创建会出错。只要将create_time放在start_time和end_time之前是方便的解决方法。对比下我修改过后的跟视频里面的sql片段:

我们可以看到在这三个字段有一个小差别,那就是给start_time,end_time,create_time三个字段都添加一个默认值,然后执行数据库语句就没问题了
这里我们需要修改下`web.xml`中的servlet版本为`3.0`
打开WEB-INF下的web.xml,修改为以下代码:

修改的原因有以下几点:高版本的Servlet支持更多的特性,更方便我们的Coding,特别是支持注解这一特性在Servlet2.3中新加入了Listener接口的实现,,我们可以使用Listener引入Spring的ContextLoaderListener举个栗子:在Servlet2.3以前我们这样配置ContextLoaderListener:
context
org.springframework.context.ContextLoaderServlet
1

在Servlet2.3以后可以使用Listener配置,也就是我们项目中使用的方法
org.springframework.context.ContextLoaderListener

两种方法的效果都是一样的,主要不要同时使用,否则会报错的
建立实体类首先建立SuccessKilled 秒杀状态表package com.suny.entity;

import java.io.Serializable;
import java.time.LocalDateTime;

public class SuccessKilled implements Serializable {
private static final long serialVersionUID = 1834437127882846202L;

private long seckillId;
/* 用户的手机号码*/
private long userPhone;
/* 秒杀的状态*/
private short state;
/* 创建时间*/
private LocalDateTime createTime;
/* 多对一,因为一件商品在库存中肯定有许多,对应的购买信息也有很多*/
private Seckill seckill;

public SuccessKilled() {
}

public SuccessKilled(long seckillId, long userPhone, short state, LocalDateTime createTime, Seckill seckill) {
this.seckillId = seckillId;
this.userPhone = userPhone;
this.state = state;
this.createTime = createTime;
this.seckill = seckill;
}

public long getSeckillId() {
return seckillId;
}

public void setSeckillId(long seckillId) {
this.seckillId = seckillId;
}

public long getUserPhone() {
return userPhone;
}

public void setUserPhone(long userPhone) {
this.userPhone = userPhone;
}

public short getState() {
return state;
}

public void setState(short state) {
this.state = state;
}

public LocalDateTime getCreateTime() {
return createTime;
}

public void setCreateTime(LocalDateTime createTime) {
this.createTime = createTime;
}

public Seckill getSeckill() {
return seckill;
}

public void setSeckill(Seckill seckill) {
this.seckill = seckill;
}

@Override
public String toString() {
return “SuccessKilled{” +
“主键ID=” + seckillId +
“, 手机号码=” + userPhone +
“, 秒杀状态=” + state +
“, 创建时间=” + createTime +
“, 秒杀的商品=” + seckill +
};
}
}
再建立Seckill 秒杀商品信息package com.suny.entity;

import java.io.Serializable;
import java.time.LocalDateTime;

public class Seckill implements Serializable {

private static final long serialVersionUID = 2912164127598660137L;
/* 主键ID*/
private long seckillId;
/* 秒杀商品名字 */
private String name;
/* 秒杀的商品编号 */
private int number;
/* 开始秒杀的时间 */
private LocalDateTime startTime;
/* 结束秒杀的时间 */
private LocalDateTime endTime;
/* 创建的时间 */
private LocalDateTime createTIme;

public Seckill() {
}

public Seckill(long seckillId, String name, int number, LocalDateTime startTime, LocalDateTime endTime, LocalDateTime createTIme) {
this.seckillId = seckillId;
this.name = name;
this.number = number;
this.startTime = startTime;
this.endTime = endTime;
this.createTIme = createTIme;
}

public long getSeckillId() {
return seckillId;
}

public void setSeckillId(long seckillId) {
this.seckillId = seckillId;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public int getNumber() {
return number;
}

public void setNumber(int number) {
this.number = number;
}

public LocalDateTime getStartTime() {
return startTime;
}

public void setStartTime(LocalDateTime startTime) {
this.startTime = startTime;
}

public LocalDateTime getEndTime() {
return endTime;
}

public void setEndTime(LocalDateTime endTime) {
this.endTime = endTime;
}

public LocalDateTime getCreateTIme() {
return createTIme;
}

public void setCreateTIme(LocalDateTime createTIme) {
this.createTIme = createTIme;
}

@Override
public String toString() {
return “com.suny.entity.Seckill{” +
“主键ID=” + seckillId +
“, 秒杀商品=” + name + \ +
“, 编号=” + number +
“, 开始秒杀时间=” + startTime +
“, 结束秒杀时间=” + endTime +
“, 创建时间=” + createTIme +
};
}
}

对实体类创建对应的mapper接口,也就是dao接口类首先创建SeckillMapper,在我这里位于com.suny.dao包下package com.suny.dao;

import com.suny.entity.Seckill;
import org.apache.ibatis.annotations.Param;

import java.time.LocalDateTime;
import java.util.List;

public interface SeckillMapper {
/**
* 根据传过来的seckillId去减少商品的库存.
*
* @param seckillId 秒杀商品ID
* @param killTime 秒杀的精确时间
* @return 如果秒杀成功就返回1,否则就返回0
*/
int reduceNumber(@Param(“seckillId”) long seckillId, @Param(“killTime”) LocalDateTime killTime);

/**
* 根据传过来的seckillId去查询秒杀商品的详情.
*
* @param seckillId 秒杀商品ID
* @return 对应商品ID的的数据
*/
Seckill queryById(@Param(“seckillId”) long seckillId);

/**
* 根据一个偏移量去查询秒杀的商品列表.
*
* @param offset 偏移量
* @param limit 限制查询的数据个数
* @return 符合偏移量查出来的数据个数
*/
List queryAll(@Param(“offset”) int offset, @Param(“limit”) int limit);
}
再创建SuccessKilledMapperpackage com.suny.dao;

import com.suny.entity.SuccessKilled;
import org.apache.ibatis.annotations.Param;

public interface SuccessKilledMapper {
/**
* 插入一条详细的购买信息.
*
* @param seckillId 秒杀商品的ID
* @param userPhone 购买用户的手机号码
* @return 成功插入就返回1, 否则就返回0
*/
int insertSuccessKilled(@Param(“seckillId”) long seckillId, @Param(“userPhone”) long userPhone);

/**
* 根据秒杀商品的ID查询SuccessKilled的明细信息.
*
* @param seckillId 秒杀商品的ID
* @param userPhone 购买用户的手机号码
* @return 秒杀商品的明细信息
*/
SuccessKilled queryByIdWithSeckill(@Param(“seckillId”) long seckillId, @Param(“userPhone”) long userPhone);
}

接下来书写xml配置文件
建立对应的`mapper.xml`
首先在src/main/resources建立com.suny.dao这个包,也就是对应mapper接口文件包一样的包名,这样符合Maven的约定,就是资源放置在Resource包下,Java包下则是放置java类文件,编译后最后还是会在同一个目录下
首先建立SeckillMapper.xml

<!–这里的

UPDATE seckill
SET number = number – 1
WHERE seckill_id = #{seckillId}
AND start_time
<![CDATA[

#{killTime}
AND end_time >= #{killTime}
AND number > 0

SELECT
*
FROM seckill AS s
WHERE s.seckill_id = #{seckillId}

SELECT
*
FROM seckill AS s
ORDER BY create_time DESC
LIMIT #{offset}, #{limit}

建立SuccessKilledMapper.xml

INSERT IGNORE INTO success_killed (seckill_id, user_phone, state)
VALUES (#{seckillId}, #{userPhone}, 0)

SELECT
sk.seckill_id,
sk.user_phone,
sk.create_time,
sk.state,
s.seckill_id “seckill.seckill_id”,
s.name “seckill.name”,
s.number “seckill”,
s.start_time “seckill.start_time”,
s.end_time “seckill.end_time”,
s.create_time “seckill.create_time”
FROM success_killed sk
INNER JOIN seckill s ON sk.seckill_id = s.seckill_id
WHERE sk.seckill_id = #{seckillId}
AND sk.user_phone= #{userPhone}

建立Mybatis的配置文件mybatis-config.xml

然后建立连接数据库的配置文件jdbc.properties,这里的属性要根据自己的需要去进行修改,切勿直接复制使用jdbc.driver=com.mysql.jdbc.Driver
jdbc.user=root
jdbc.password=root
jdbc.url=jdbc:mysql://localhost:3306/seckill?useUnicode=true&characterEncoding=utf-8
建立Spring的dao的配置文件,在resources包下创建applicationContext-dao.xml

基础的部分我们搭建完成了,然后要开始测试了 在IDEA里面有一个快速建立测试的快捷键Ctrl+Shift+T,在某个要测试的类里面按下这个快捷键就会出现Create new Test,然后选择你要测试的方法跟测试的工具就可以了,这里我们使用Junit作为测试建立SeckillMapperTest文件,代码如下package com.suny.dao;

import com.suny.entity.Seckill;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

import javax.annotation.Resource;

import java.time.LocalDateTime;
import java.util.List;

import static org.junit.Assert.*;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration({“classpath:spring/applicationContext-dao.xml”})
public class SeckillMapperTest {
@Resource
private SeckillMapper seckillMapper;

@Test
public void reduceNumber() throws Exception {
long seckillId=1000;
LocalDateTime localDateTime=LocalDateTime.now();
int i = seckillMapper.reduceNumber(seckillId, localDateTime);
System.out.println(i);
}

@Test
public void queryById() throws Exception {
long seckillId = 1000;
Seckill seckill = seckillMapper.queryById(seckillId);
System.out.println(seckill.toString());
}

@Test
public void queryAll() throws Exception {
List seckills = seckillMapper.queryAll(0, 100);
for (Seckill seckill : seckills) {
System.out.println(seckill.toString());
}
}

}

测试中可能会出现Mybatis参数绑定失败的错误,在mapper接口中的方法里面添加@Param的注解,显示的告诉mybatis参数的名称是什么,例如
List queryAll(@Param(“offset”) int offset, @Param(“limit”) int limit);

PS:如果觉得我的分享不错,欢迎大家随手点赞、转发。
来源:http://t.cn/EoOnJbR

创业项目群,学习操作 18个小项目,添加 微信:923199819  备注:小项目

本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 zoodoho@qq.com举报,一经查实,本站将立刻删除。
如若转载,请注明出处:https://www.zodoho.com/140394.html