设计SpringBoot目的是用来简化Spring应用的初始搭建以及开发过程
①创建新模块,选择Spring Initializr,并配置模块相关基础信息
②:选择当前模块需要使用的技术集
③:开发控制器类
package com.itheima.test.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* @Description: // 类说明
* @ClassName: BookController // 类名
* @Author: 曾伟鸿 // 创建者
* @Date: 2022/2/6 15:08 // 时间
* @Version: 1.0 // 版本
*/
@RestController
@RequestMapping("/books")
public class BookController {
@GetMapping
public String getById(){
System.out.println("66666");
return "66666666666";
}
}
④:运行自动生成的Application类
- Spring程序缺点
- 依赖设置繁琐
- 配置繁琐
- SpringBoot程序优点
- 起步依赖(简化依赖配置)
- 自动配置(简化常用工程相关配置)
- 辅助功能(内置服务器,……)
注意事项:基于idea开发SpringBoot程序需要确保联网且能够加载到程序框架结构
- 基于Idea创建SpringBoot工程
- 基于官网创建SpringBoot工程
- 基于阿里云创建SpringBoot工程
- 手工创建Maven工程修改为SpringBoot工程(一般不用)
①基于Idea创建SpringBoot工程
上面第一个创建工程就是基于idea创建的SpringBoot工程
②基于官网创建SpringBoot工程
步骤:
③基于阿里云创建SpringBoot工程
基于阿里云创建项目,地址:https://start.aliyun.com
注意事项:
- 阿里云提供的坐标版本较低,如果需要使用高版本,进入工程后手工切换SpringBoot版本
- 阿里云提供的工程模板与Spring官网提供的工程模板略有不同
步骤:
- Setting → File Types → Ignored Files and Folders
- 输入要隐藏的文件名,支持*号通配符
- 回车确认添加
开发SpringBoot程序要继承spring-boot-starter-parent
2. spring-boot-starter-parent中定义了若干个依赖管理
3. 继承parent模块可以避免多个依赖使用相同技术时出现依赖版本冲突
4. 继承parent的形式也可以采用引入依赖的形式实现效果
1.开发SpringBoot程序需要导入坐标时通常导入对应的starter
2. 每个不同的starter根据功能不同,通常包含多个依赖坐标
3. 使用starter可以实现快速配置的效果,达到简化配置的目的
- starter
- SpringBoot中常见项目名称,定义了当前项目使用的所有依赖坐标,以达到减少依赖配置的目的
- parent
- 所有SpringBoot项目要继承的项目,定义了若干个坐标版本号(依赖管理,而非依赖),以达到减少依赖冲突的目的
- spring-boot-starter-parent各版本间存在着诸多坐标版本不同
- 实际开发
- 使用任意坐标时,仅书写GAV中的G和A,V由SpringBoot提供,除非SpringBoot未提供对应版本V
- 如发生坐标错误,再指定Version(要小心版本冲突)
1.SpringBoot工程提供引导类用来启动程序
2. SpringBoot工程启动后创建并初始化Spring容器
package com.itheima.test;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
@SpringBootApplication
public class TestApplication {
public static void main(String[] args) {
SpringApplication.run(TestApplication.class, args);
}
}
1. 内嵌Tomcat服务器是SpringBoot辅助功能之一
2. 内嵌Tomcat工作原理是将Tomcat服务器作为对象运行,并将该对象交给Spring容器管理
3. 变更内嵌服务器思想是去除现有服务器,添加全新的服务器
内置服务器
使用maven依赖管理变更起步依赖项
org.springframework.boot
spring-boot-starter-web
org.springframework.boot
spring-boot-starter-tomcat
org.springframework.boot
spring-boot-starter-jetty
①修改服务器端口
server.port=80
②关闭运行日志图标(banner)
spring.main.banner-mode=off
③设置日志相关
logging.level.root=debug
- properties(传统格式/默认格式)
server.port=80
- yml(主流格式)
server: port: 81
- yaml
server: port: 82
SpringBoot配置文件加载顺序:
application.properties > application.yml > application.yaml
常用配置文件种类:
application.yml
不同配置文件中相同配置按照加载优先级相互覆盖,不同配置文件中不同配置全部保留
- Setting → Project Structure → Facets
- 选中对应项目/工程
- Customize Spring Boot
- 选择配置文件
- YAML(YAML Ain't Markup Language),一种数据序列化格式
- 优点:
- 容易阅读
- 容易与脚本语言交互
- 以数据为核心,重数据轻格式
- YAML文件扩展名
- .yml(主流)
- .yaml
- 大小写敏感
- 属性层级关系使用多行描述,每行结尾使用冒号结束
- 使用缩进表示层级关系,同层级左侧对齐,只允许使用空格(不允许使用Tab键)
- 属性值前面添加空格(属性名与属性值之间使用冒号+空格作为分隔)
- # 表示注释
- 核心规则:数据前面要加空格与冒号隔开
在属性名书写位置的下方使用减号作为数据开始符号,每行书写一个数据,减号与数据间空格分隔
使用@Value读取单个数据,属性名引用方式:${一级属性名.二级属性名……}
- 使用@Value配合SpEL读取单个数据
- 如果数据存在多层级,依次书写层级名称即可
- 在配置文件中可以使用${属性名}方式引用属性值
- 如果属性中出现特殊字符,可以使用双引号包裹起来作为字符解析
- 使用Environment对象封装全部配置信息
- 使用@Autowired自动装配数据到Environment对象中
- 使用@ConfigurationProperties注解绑定配置信息到封装类中
- 封装类需要定义为Spring管理的bean,否则无法进行属性注入
整合第三方技术通用方式
- 导入对应的starter
- 根据提供的配置格式,配置非默认值对应的配置项
- 导入测试对应的starter
- 测试类使用@SpringBootTest修饰
- 使用自动装配的形式添加要测试的对象
- 测试类如果存在于引导类所在包或子包中无需指定引导类 2
- 测试类如果不存在于引导类所在的包或子包中需要通过classes 属性指定引导类
①导入测试对应的starter
org.springframework.boot
spring-boot-starter-test
test
org.junit.vintage
junit-vintage-engine
②测试类使用@SpringBootTest修饰
- 名称:@SpringBootTest
- 类型:测试类注解
- 位置:测试类定义上方
- 作用:设置JUnit加载的SpringBoot启动类
- 相关属性
- classes:设置SpringBoot启动类
- 范例:
@SpringBootTest(classes = Springboot05JUnitApplication.class) class Springboot07JUnitApplicationTests {}
注意事项:如果测试类在SpringBoot启动类的包或子包中,可以省略启动类的设置,也就是省略classes的设定
③使用自动装配的形式添加要测试的对象
@SpringBootTest
class Springboot07JunitApplicationTests {
@Autowired
private BookService bookService;
@Test
public void testSave(){
bookService.save();
}
}
- 勾选MyBatis技术,也就是导入MyBatis对应的starter
- MyBatis模块需要使用的技术集(MyBatis、MySQL)
- 数据库连接相关信息转换成配置
- 数据库SQL映射需要添加@Mapper被容器识别到
- MySQL 8.X驱动强制要求设置时区
- 修改url,添加serverTimezone设定
- 修改MySQL数据库配置(略)
- 驱动类过时,提醒更换为com.mysql.cj.jdbc.Driver
②设置数据源参数
SpringBoot版本低于2.4.3(不含),Mysql驱动版本大于8.0时,需要在url连接串中配置时区
jdbc:mysql://localhost:3306/mabatis-puls?serverTimezone=UTC
或在MySQL数据库端配置时区解决此问题
spring:
datasource:
url: jdbc:mysql://localhost:3306/mabatis-puls?serverTimezone=UTC
driver-class-name: com.mysql.cj.jdbc.Driver
username: *******
password: ******
③定义数据层接口与映射配置
@Mapper
public interface UserDao {
@Select("select * from user")
public List getAll();
}
④测试类中注入dao接口,测试功能
@SpringBootTest
class Springboot08MybatisApplicationTests {
@Autowired
private BookDao bookDao;
@Test
public void testGetById() {
Book book = bookDao.getById(1);
System.out.println(book);
}
}
⑤
MyBatis-Plus与MyBatis区别
- 导入坐标不同
- 数据层实现简化
步骤:
- 手工添加MyBatis-Plus对应的starter
- 数据层接口使用BaseMapper简化开发
- 需要使用的第三方技术无法通过勾选确定时,需要手工添加坐标
①手动添加SpringBoot整合MyBatis-Plus的坐标,可以通过mvnrepository获取
由于SpringBoot中未收录MyBatis-Plus的坐标版本,需要指定对应的Version(用官方网站建的模块),如果用阿里云的就不用导入了
com.baomidou
mybatis-plus-boot-starter
3.4.3
②由于SpringBoot中未收录MyBatis-Plus的坐标版本,需要指定对应的Version
@Mapper
public interface UserDao extends BaseMapper {
}
③其他同SpringBoot整合MyBatis
1. 整合Druid需要导入Druid对应的starter
2. 根据Druid提供的配置方式进行配置
- 实体类开发————使用Lombok快速制作实体类
- Dao(数据层)开发————整合MyBatisPlus,制作数据层测试类
- Service开发————基于MyBatisPlus进行增量开发,制作业务层测试类
- Controller开发————基于Restful开发,使用PostMan测试接口功能
- Controller开发————前后端开发协议制作
- 页面开发————基于VUE+ElementUI制作,前后端联调,页面数据处理,页面消息处理 列表、新增、修改、删除、分页、查询
- 项目异常处理
- 按条件查询————页面功能调整、Controller修正功能、Service修正功能
- Lombok,一个Java类库,提供了一组注解,简化POJO实体类开发
org.projectlombok lombok
- lombok版本由SpringBoot提供,无需指定版本
开发常用三个注解:
- @Data
- @NoArgsConstructor
- @AllArgsConstructor
为当前实体类在编译期设置对应的get/set方法,toString方法,hashCode方法,equals方法等
例如:
package com.itheima2.pojo;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Book {
private int id;
private String name;
private String author;
private Integer price;
private int sales;
private int stock;
private String img_path;
}
- 手工导入starter坐标(2个)
- 配置数据源与MyBatisPlus对应的配置(为方便调试可以开启MyBatisPlus的日志)
- 开发Dao接口(继承BaseMapper)
- 制作测试类测试Dao功能是否有效
com.baomidou
mybatis-plus-boot-starter
3.4.3
com.alibaba
druid-spring-boot-starter
1.2.6
#配置数据源
spring:
datasource:
druid:
driver-class-name: com.mysql.cj.jdbc.Driver
url: jdbc:mysql://localhost:3306/book?servierTimezone=UTC
username: root
password: Zwh1174946082!
#Mp配置
mybatis-plus:
global-config:
db-config:
#设置前缀
table-prefix: t_
#设置递增类型
id-type: auto
#开启Mp运行日志
configuration:
log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
package com.itheima2.dao;
import com.baomidou.mybatisplus.core.mapper.BaseMapper;
import com.itheima2.pojo.Book;
import org.apache.ibatis.annotations.Mapper;
@Mapper
public interface BookDao extends BaseMapper {
}
- 使用IPage封装分页数据
- 分页操作依赖MyBatisPlus分页拦截器实现功能
- 借助MyBatisPlus日志查阅执行SQL语句
IPage对象中封装了分页操作中的所有数据
- 当前页码值 page.getTotal()
- 每页显示数据数量 page.getSize()
- 最大页码值
- 数据总量
@Test
void testGetPage(){
IPage page = new Page(1,5);
bookDao.selectPage(page,null);
}
package com.itheima2.config;
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class MPConfig {
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor() {
//1.创建一个MP的拦截器
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
//2.添加具体的拦截器
//添加分页的拦截器
interceptor.addInnerInterceptor(new PaginationInnerInterceptor());
return interceptor;
}
}
- 推荐使用LambdaQueryWrapper对象,所有查询操作封装成方法调用(这样就不用手动去写属性了)
- 所有查询操作封装成方法调用
- 查询条件支持动态条件拼装
@Test
void testGetByCondition(){
IPage page = new Page(1,10);
LambdaQueryWrapper lqw = new LambdaQueryWrapper();
lqw.like(Book::getName,"Spring");
bookDao.selectPage(page,lqw);
}
快速开发方案
- 使用MyBatisPlus提供有业务层通用接口(ISerivce)与业务层通用实现类(ServiceImpl)
- 在通用类基础上做功能重载或功能追加
- 注意重载时不要覆盖原始操作,避免原始提供的功能丢失
public interface IBookService extends IService {
//追加的功能
//可以用@Override查看是否与IService里面的方法重名,如果@Override不报错就是重名了,报错就不重名
}
@Service
public class BookServiceImpl2 extends ServiceImpl implements IBookService {
//追加的功能
@Autowired
private BookDao bookDao;
public Boolean insert(Book book) {
return bookDao.insert(book) > 0;
}
public Boolean modify(Book book) {
return bookDao.updateById(book) > 0;
}
public Boolean delete(Integer id) {
return bookDao.deleteById(id) > 0;
}
public Book get(Integer id) {
return bookDao.selectById(id);
}
}
- 基于Restful制作表现层接口
- 新增:POST
- 删除:DELETE
- 修改:PUT
- 查询:GET
- 接收参数
- 实体数据:@RequestBody
- 路径变量:@PathVariable
package com.itheima.controller;
import com.baomidou.mybatisplus.core.metadata.IPage;
import com.itheima.controller.utils.R;
import com.itheima.domain.Book;
import com.itheima.service.IBookService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;
import java.io.IOException;
import java.util.List;
@RestController
@RequestMapping("/books")
public class BookController {
@Autowired
private IBookService bookService;
@GetMapping
public R getAll() {
return new R(true, bookService.list());
}
@PostMapping
public R save(@RequestBody Book book) throws IOException {
//测试用的专门用来制作异常
if (book.getName().equals("123")) throw new IOException();
boolean flag = bookService.save(book);
return new R(flag, flag ? "添加成功^_^" : "添加失败-_-!");
}
@PutMapping
public R update(@RequestBody Book book) throws IOException {
if (book.getName().equals("123")) throw new IOException();
boolean flag = bookService.modify(book);
return new R(flag, flag ? "修改成功^_^" : "修改失败-_-!");
}
@DeleteMapping("{id}")
public R delete(@PathVariable Integer id) {
return new R(bookService.delete(id));
}
@GetMapping("{id}")
public R getById(@PathVariable Integer id) {
return new R(true, bookService.getById(id));
}
@GetMapping("{currentPage}/{pageSize}")
public R getPage(@PathVariable int currentPage, @PathVariable int pageSize, Book book) {
IPage page = bookService.getPage(currentPage, pageSize, book);
//如果当前页码值大于了总页码值,那么重新执行查询操作,使用最大页码值作为当前页码值
if (currentPage > page.getPages()) {
page = bookService.getPage((int) page.getPages(), pageSize, book);
}
return new R(true, page);
}
}
设计表现层返回结果的模型类,用于后端与前端进行数据格式统一,也称为前后端数据协议
- 设计统一的返回值结果类型便于前端开发读取数据
- 返回值结果类型可以根据需求自行设定,没有固定格式
- 返回值结果模型类用于后端与前端进行数据格式统一,也称为前后端数据协议
package com.itheima2.controller.utils;
import lombok.Data;
@Data
public class R {
private Boolean flag;
private Object data;
private String msg;
public R() {
}
public R(Boolean flag) {
this.flag = flag;
}
public R(Boolean flag, Object data) {
this.flag = flag;
this.data = data;
}
public R(Boolean flag, String msg) {
this.flag = flag;
this.msg = msg;
}
public R(String msg) {
this.flag = false;
this.msg = msg;
}
}
- 单体项目中页面放置在resources/static目录下
- created钩子函数用于初始化页面时发起调用
- 页面使用axios发送异步请求获取数据后确认前后端是否联 通