如果觉得不错就点击右上角star鼓励一下笔者吧(#^.^#)
技术栈
- 后端: SpringBoot2.x + Mybatis
- 前端: Vue.JS2.x + ElementUI
测试环境
- IDEA + SpringBoot-2.0.5
启动说明
启动前,请配置好 application.yml 中连接数据库的用户名和密码,以及Redis服务器的地址和端口信息。
启动前,请创建数据库
seckill,建表SQL语句放在:/db/sys_schema.sql。具体的建表和建库语句请仔细看SQL文件。配置完成后,运行位于
src/main/cn/tycoding/下的SpringbootApplication中的main方法,访问http://localhost:8080/进行API测试。
项目设计
. ├── README ├── README.md ├── db ├── mvnw ├── mvnw.cmd ├── pom.xml ├── spring-boot.iml ├── src │ ├── main │ │ ├── java │ │ │ └── cn │ │ │ └── tycoding │ │ │ ├── SpringbootApplication.java -- Spring Boot启动器类 │ │ │ ├── controller -- MVC-WEB层 │ │ │ ├── entity -- 实体类 │ │ │ ├── interceptor -- 自定义拦截器 │ │ │ ├── mapper -- mybatis-Mapper层映射接口,或称为DAO层 │ │ │ └── service -- service业务层 │ │ └── resources -- Spring Boot资源文件目录 │ │ ├── application.yml -- Spring Boot核心配置文件 │ │ ├── mapper -- Mybatis Mapper层XML配置文件 │ │ ├── static -- 前端静态文件(主要是JS、CSS、Image文件,一般不放HTML页面) │ │ │ ├── css │ │ │ ├── image │ │ │ ├── js │ │ │ ├── lib │ │ └── templates -- Thymeleaf模板引擎识别的HTML页面目录,存放HTML页面(相当于之前的WEB—INF目录,即不能通过浏览器直接访问) │ └── test 更多文档将在我的公众号 程序员涂陌 中陆续发布,请持续关注!
| 程序员涂陌 |
|---|
![]() |
| Alipay | WechatPay |
|---|---|
![]() | ![]() |
开始实战Spring Boot项目,首先,你需要将Spring Boot工程搭建出来。
Spring Boot工程搭建请看我的博客:Spring Boot入门之工程搭建
Spring Boot提供了很多应用启动器,分别用来支持不同的功能,说白了就是pom.xml中的依赖配置,因为Spring Boot的自动化配置特性,我们并不需再考虑项目依赖版本问题,使用Spring Boot的应用启动器,它能自动帮我们将相关的依赖全部导入到项目中。
我们这里介绍几个常见的应用启动器:
spring-boot-starter: Spring Boot的核心启动器,包含了自动配置、日志和YAMLspring-boot-starter-aop: 支持AOP面向切面编程的功能,包括spring-aop和AspecJspring-boot-starter-cache: 支持Spring的Cache抽象spring-boot-starter-artermis: 通过Apache Artemis支持JMS(Java Message Service)的APIspring-boot-starter-data-jpa: 支持JPAspring-boot-starter-data-solr: 支持Apache Solr搜索平台,包括spring-data-solrspring-boot-starter-freemarker: 支持FreeMarker模板引擎spring-boot-starter-jdbc: 支持JDBC数据库spring-boot-starter-Redis: 支持Redis键值储存数据库,包括spring-redisspring-boot-starter-security: 支持spring-securityspring-boot-starter-thymeleaf: 支持Thymeleaf模板引擎,包括与Spring的集成spring-boot-starter-web: 支持全栈式web开发,包括tomcat和Spring-WebMVCspring-boot-starter-log4j: 支持Log4J日志框架spring-boot-starter-logging: 引入Spring Boot默认的日志框架Logback
Spring Boot项目(即Maven项目),当然拥有最基础的Maven项目结构。除此之外:
Spring Boot项目中不包含webapp(webroot)目录。
Spring Boot默认提供的静态资源目录需要置于classpath下,且其下的目录名称要符合一定规定。
Spring Boot默认不提倡用XML配置文件,主张使用YML作为配置文件格式,YML有更简洁的语法。当然也可以使用.properties作为配置文件格式。
Spring Boot官方推荐使用Thymeleaf作为前端模板引擎,并且Thymeleaf默认将templates作为静态页面的存放目录(由配置文件指定)。
Spring Boot默认将
resources作为静态资源的存放目录,存放前端静态文件、项目配置文件。Spring Boot规定
resources下的子级目录名要符合一定规则,一般我们设置resources/static为前端静态(JS,CSS)的存放目录;设置resources/templates作为HTML页面的存放目录。Spring Boot指定的Thymeleaf模板引擎文件目录
/resources/templates是受保护的目录,这与之前的WEB项目的WEB-INF文件夹和类似,特点就是里面的静态资源不能直接访问,一般我们通过Controller映射访问。建议将Mybatis-Mapper的XML映射文件放于
resources/目录下,我这里设为resources/mapper目录,且src/main/java/Mapper下的Mapper层接口要使用@Mapper注解标识,不然mybatis找不到接口对应的XML映射文件。SpringBootApplication.java为项目的启动器类,项目不需要部署到Tomcat上,由SpringBoot提供的服务器部署项目(运行启动器类即可);且SpringBoot会自动扫描该启动器同级和子级下用注解标识的Bean,也就是之前我们配置的<context:component-scan>在SpringBoot中是完全不需类似的配置的。Spring Boot不建议使用JSP页面,如果想使用,请自行百度解决办法。我们常用HTML页面+Thymeleaf模板引擎。Thyemeleaf模板引擎提供了很多内置语法,比如:可以通过
<div th:text="${xx}">取出来后端存放在域对象中的数据。上面说了Spring Boot提供的存放HTML静态页面的目录
resources/templates是受保护的目录,访问其中的HTML页面要通过Controller映射,这就间接规定了你需要配置Spring的视图解析器,且Controller类不能使用@RestController标识。
首先:我想特殊强调的是:SpringBoot不是对Spring功能上的增强,而是提供了一种快速使用Spring的方式。一定要切记这一点。
学习SpringBoot框架,只是为了更简便的使用Spring框架,我们在SSM阶段学习的知识现在放在Spring Boot框架上开发是完全适用的,我们学习的大多数是SpringBoot的自动化配置方式。
因为Spring Boot框架的一大优势就是自动化配置,从pom.xml的配置中就能明显感受到。
所以这里推荐一下我之前的SSM阶段整合项目: SSM详细入门整合案例SSM+Redis+Shiro+Solr+Vue.js整合项目
本项目的依赖文件请看Github仓库:spring-boot/pom.xml
本项目数据库表设计请看GitHub仓库:spring-boot/db/
请运行项目前,先把数据库表结构建好
之前已经说过:SpringBoot框架不是对Spring功能上的增强,而是提供了一种快速使用Spring的方式
所以说,SpringBoot整合Mybatis的思想和Spring整合Mybatis的思想基本相同,不同之处有两点:
1.Mapper接口的XML配置文件变化。之前我们使用Mybatis接口代理开发,规定Mapper映射文件要和接口在一个目录下;而这里Mapper映射文件置于
resources/mapper/下,且置于src/main/java/下的Mapper接口需要用@Mapper注解标识,不然映射文件与接口无法匹配。2.SpringBoot建议使用YAML作为配置文件,它有更简便的配置方式。所以整合Mybatis在配置文件上有一定的区别,但最终都是那几个参数的配置。
关于YAML的语法请自行百度,我这里也仅仅是满足基本的配置需求,不涉及那种不易理解的语法。
本例详细代码请看GitHub仓库:spring-boot/resources/application.yml
在Spring阶段用XML配置mybatis无非就是配置:1.连接池;2.数据库url连接;3.mysql驱动;4.其他初始化配置
spring: datasource: name: springboottype: com.alibaba.druid.pool.DruidDataSource#druid相关配置druid: #监控统计拦截的filtersfilter: stat#mysql驱动driver-class-name: com.mysql.jdbc.Driver#基本属性url: jdbc:mysql://127.0.0.1:3306/springboot?useUnicode=true&characterEncoding=UTF-8&allowMultiQueries=trueusername: rootpassword: root#配置初始化大小/最小/最大initial-size: 1min-idle: 1max-active: 20#获取连接等待超时时间max-wait: 60000#间隔多久进行一次检测,检测需要关闭的空闲连接time-between-eviction-runs-millis: 60000#mybatis配置mybatis: mapper-locations: classpath:mapper/*.xmltype-aliases-package: cn.tycoding.entity注意:空格代表节点层次;注释部分用#标记
解释
我们实现的是spring-mybatis的整合,包含mybatis的配置以及datasource数据源的配置当然属于spring配置中的一部分,所以需要在
spring:下。mapper-locations相当于XML中的<property name="mapperLocations">用来扫描Mapper层的配置文件,由于我们的配置文件在resources下,所以需要指定classpath:。type-aliases-package相当与XML中<property name="typeAliasesPackase">别名配置,一般取其下实体类类名作为别名。datasource数据源的配置,name表示当前数据源的名称,类似于之前的<bean id="dataSource">id属性,这里可以任意指定,因为我们无需关注Spring是怎么注入这个Bean对象的。druid代表本项目中使用了阿里的druid连接池,driver-class-name:相当于XML中的<property name="driverClassName">;url代表XML中的<property name="url">;username代表XML中的<property name="username">;password代表XML中的<property name="password">;其他druid的私有属性配置不再解释。这里注意druid连接池和c3p0连接池在XML的的name中就不同,在此处SpringBoot的配置中当然名称也不同。
如果Spring整合Mybtis的配置你已经很熟悉了,那么这个配置你肯定也很眼熟,从英文名称上就很容易区分出来。这里需要注意的就是YAML语法规定不同行空格代表了不同的层级结构。
既然完成了SpringBoot-Mybatis基本配置下面我们实战讲解如何实现基本的CRUD。
1.在
src/main/java/cn/tycoding/entity/下新建User.java实体类
publicclassUserimplementsSerializable{privateLongid; //编号privateStringusername; //用户名privateStringpassword; //密码//getter/setter }2.在
src/main/java/cn/tycoding/service/下创建BaseService.java通用接口,目的是简化service层接口基本CRUD方法的编写。
publicinterfaceBaseService<T>{// 查询所有List<T> findAll(); //根据ID查询List<T> findById(Longid); //添加voidcreate(Tt); //删除(批量)voiddelete(Long... ids); //修改voidupdate(Tt)}以上就是我对Service层基本CRUD接口的简易封装,使用了泛型类,其继承接口指定了什么泛型,T就代表什么类。
3.在
src/main/java/cn/tycoding/service/下创建UserService.java接口:
publicinterfaceUserServiceextendsBaseService<User>{}4.在
src/main/java/cn/tycoding/service/impl/下创建UserServiceImpl.java实现类:
@ServicepublicclassUserServiceImplimplementsUserService{@AutowiredprivateUserMapperuserMapper; @OverridepublicList<User> findAll(){returnuserMapper.findAll()} //其他方法省略 }5.在
src/main/java/cn/tycoding/mapper/下创建UserMapper.javaMapper接口类:
@MapperpublicinterfaceUserMapper{List<User> findAll()}如上,我们一定要使用@Mapper接口标识这个接口,不然Mybatis找不到其对应的XML映射文件。
6.在
src/main/resources/mapper/下创建UserMapper.xml映射文件:
<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPEmapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd" > <mappernamespace="cn.tycoding.mapper.UserMapper"> <!-- 查询所有 --> <selectid="findAll"resultType="cn.tycoding.entity.User"> SELECT * FROM tb_user </select> </mapper>7.在
src/main/java/cn/tycoding/controller/admin/下创建UserController.java
@RestControllerpublicclassUserController{@AutowiredprivateUserServiceuserService; @RequestMapping("/findAll") publicList<User> findAll(){returnuserService.findAll()} }8.运行
src/main/java/cn/tycoding/SpringbootApplication.java的main方法,启动springboot
在浏览器上访问localhost:8080/findAll即可得到一串JSON数据。
看了上面一步步的讲解。你应该明白了,其实和SSM阶段的CRUD基本相同,这里我就不再举例其他方法。
下面我们讲解一下不同的地方:
因为Thymeleaf指定的目录src/main/resources/templates/是受保护的目录,其下的资源不能直接通过浏览器访问,可以使用Controller映射的方式访问,怎么映射呢?
1.在application.yml中添加配置
spring: thymeleaf: prefix: classpath:/templates/check-template-location: truesuffix: .htmlencoding: UTF-8mode: LEGACYHTML5cache: false指定Thymeleaf模板引擎扫描resources下的templates文件夹中已.html结尾的文件。这样就实现了MVC中关于视图解析器的配置:
<!-- 配置视图解析器 --> <beanclass="org.springframework.web.servlet.view.InternalResourceViewResolver"> <propertyname="prefix"value="/"/> <propertyname="suffix"value=".jsp"/> </bean>是不是感觉方便很多呢?但这里需要注意的是:classpath:后的目录地址一定要先加/,比如目前的classpath:/templates/。
2.在Controller添加映射方法
@GetMapping(value ={"/", "/index"}) publicStringindex(){return"home/index"}这样,访问localhost:8080/index将直接跳转到resources/templates/home/index.html页面。
首先我们需要在application.yml中配置pageHelper插件
pagehelper: pagehelperDialect: mysqlreasonable: truesupport-methods-arguments: true我这里使用了Mybatis的PageHelper分页插件,前端使用了ElementUI自带的分页插件:具体的教程请查看我的博客:SpringMVC+ElementUI实现分页查询
核心配置:
UserServiceImp.java
publicPageBeanfindByPage(Goodsgoods, intpageCode, intpageSize){//使用Mybatis分页插件PageHelper.startPage(pageCode, pageSize); //调用分页查询方法,其实就是查询所有数据,mybatis自动帮我们进行分页计算Page<Goods> page = goodsMapper.findByPage(goods); returnnewPageBean(page.getTotal(), page.getResult())}这里涉及的无非就是SpringMVC的文件上传,详细的教程请参看我的博客:SpringMVC实现文件上传和下载
因为本项目中前端使用了ElementUI+Vue.JS技术,所以前端的文件上传和回显教程请看我的博客:SpringMVC+ElementUI实现图片上传和回显
除了代码的编写,这里还要在application.yml中进行配置:
spring: servlet: multipart: max-file-size: 10Mbmax-request-size: 100Mb这就相当于SpringMVC的XML配置:
<beanid="multipartResolver"class="org.springframework.web.multipart.commons.CommonsMultipartResolver"> <propertyname="maxUploadSize"value="500000"/> </bean>本项目,我们先不整合Shiro和Spring Security这些安全框架,使用Spring AOP切面编程思想实现简单的登录拦截:
@Component@AspectpublicclassMyInterceptor{@Pointcut("within (cn.tycoding.controller..*) && !within(cn.tycoding.controller.admin.LoginController)") publicvoidpointCut(){} @Around("pointCut()") publicObjecttrackInfo(ProceedingJoinPointjoinPoint) throwsThrowable{ServletRequestAttributesattributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes(); HttpServletRequestrequest = attributes.getRequest(); Useruser = (User) request.getSession().getAttribute("user"); if (user == null){attributes.getResponse().sendRedirect("/login"); //手动转发到/login映射路径 } returnjoinPoint.proceed()} }解释
关于Spring AOP的切面编程请自行百度,或者你也可以看我的博客:Spring AOP思想。我们需要注意以下几点
一定要熟悉AspectJ的切点表达式,在这里:
..*表示其目录下的所有方法和子目录方法。如果进行了登录拦截,即在session中没有获取到用户的登录信息,我们可能需要手动转发到
login页面,这里访问的是login映射。基于2,一定要指定Object返回值,若AOP拦截的Controller return了一个视图地址,那么本来Controller应该跳转到这个视图地址的,但是被AOP拦截了,那么原来Controller仍会执行return,但是视图地址却找不到404了。
切记一定要调用proceed()方法,proceed():执行被通知的方法,如不调用将会阻止被通知的方法的调用,也就导致Controller中的return会404。
如果大家有兴趣,欢迎大家加入我的Java交流群:671017003 ,一起交流学习Java技术。博主目前一直在自学JAVA中,技术有限,如果可以,会尽力给大家提供一些帮助,或是一些学习方法,当然群里的大佬都会积极给新手答疑的。所以,别犹豫,快来加入我们吧!
If you have some questions after you see this article, you can contact me or you can find some info by clicking these links.






