SpringBoot SSMP 整合案例分享
作者:Lemonade22
这篇文章主要介绍了SpringBoot SSMP 整合案例分享,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
前言:
- - 先开发基础CRUD功能,做一层测一层
- - 调通页面,确认异步提交成功后,制作所有功能
- - 添加分页功能与查询功能
1 搭建SpringBoot应用
- 勾选 SpringMVC 与 MySQL 坐标
- 修改配置文件为yml格式
- 设置端口为80方便访问(可选)
2 实体类开发
Lombok,一个Java类库,提供了一组注解,简化POJO实体类开发
- lombok版本由SpringBoot提供,无需指定版本。
- 常用注解:@Data
- 为当前实体类在编译期设置对应的 get/set 方法,toString方法,hashCode方法,equals方法等
@Data public class Book { private Integer id; private String type; private String name; private String description; }
3 数据层(dao层)开发
技术实现方案:
- MyBatisPlus
- Druid
- (1)导入 MyBatisPlus 与 Druid 对应的 starter
- (2)配置数据源与 MyBatisPlus 对应的基础配置(id 生成策略使用数据库自增策略)
spring: datasource: druid: driver-class-name: com.mysql.cj.jdbc.Driver url: jdbc:mysql://localhost:3306/ssm_db?servierTimezone=UTC username: root password: root mybatis-plus: global-config: db-config: table-prefix: tbl_ id-type: auto
(3)继承 BaseMapper 并指定泛型
@Mapper public interface BookDao extends BaseMapper<Book> { }
(4)制作测试类测试结果
@SpringBootTest public class BookDaoTestCase { @Autowired private BookDao bookDao; @Test void testSave(){ Book book = new Book(); book.setName("测试数据"); book.setType("测试类型"); bookDao.insert(book); } @Test void testGetById() { System.out.println(bookDao.selectById(1)); } }
(5)为方便调试可以开启 MyBatisPlus 的日志(使用配置方式开启日志,设置日志输出方式为标准输出)
mybatis-plus: global-config: db-config: table-prefix: tbl_ id-type: auto configuration: log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
4 数据层开发分页功能
- 分页操作需要设定分页对象
IPage
,IPage 对象中封装了分页操作中的所有数据(数据、当前页码值、每页数据总量、最大页码值、数据总量)。 - 分页操作是在 MyBatisPlus 的常规操作基础上增强得到,内部是动态的拼写 SQL 语句,使用 MyBatisPlus 拦截器实现。
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; // MyBatisPlus拦截器 @Configuration public class MPConfig { @Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { //1.定义Mp拦截 MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor(); //2.添加具体的拦截器 interceptor.addInnerInterceptor(new PaginationInnerInterceptor()); return interceptor; } }
5 数据层开发☞条件查询功能 QueryWrapper
- 使用
QueryWrapper
对象封装查询条件,推荐使用LambdaQueryWrapper
对象,将所有查询操作封装成方法调用。
// 条件查询功能 @Test void testGetByCondition(){ IPage page = new Page(1,10); LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>(); lqw.like(Book::getName,"Spring"); bookDao.selectPage(page,lqw); } @Test void testGetByCondition2(){ QueryWrapper<Book> qw = new QueryWrapper<Book>(); qw.like("name","Spring"); bookDao.selectList(qw); }
- 支持动态拼写查询条件
Strings.isNotEmpty(name)
@Test void testGetByCondition(){ String name = "Spring"; IPage page = new Page(1,10); LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>(); lqw.like(Strings.isNotEmpty(name),Book::getName,"Spring"); bookDao.selectPage(page,lqw); }
6 业务层(Service层)开发
# Service层接口定义与数据层接口定义具有较大区别,不要混用 // 业务层关注的是业务操作 login(String username,String password); // 数据层关注的是数据库操作 selectByUserNameAndPassword(String username,String password);
// 接口定义 public interface BookService { boolean save(Book book); boolean delete(Integer id); boolean update(Book book); Book getById(Integer id); List<Book> getAll(); IPage<Book> getByPage(int currentPage,int pageSize); }
// 实现类定义 @Service public class BookServiceImpl implements BookService { @Autowired private BookDao bookDao; public Boolean save(Book book) { return bookDao.insert(book) > 0; } public Boolean delete(Integer id) { return bookDao.deleteById(id) > 0; } public Boolean update(Book book) { return bookDao.updateById(book) > 0; } public Book getById(Integer id) { return bookDao.selectById(id); } public List<Book> getAll() { return bookDao.selectList(null); } public IPage<Book> getByPage(int currentPage, int pageSize) { IPage page = new Page<Book>(currentPage,pageSize); return bookDao.selectPage(page,null); } }
7 业务层开发——快速开发☞使用ISerivce和ServiceImpl
- > - 快速开发方案
- > - 使用MyBatisPlus提供的业务层通用接口(ISerivce<T>)与业务层通用实现类(`ServiceImpl<M,T>`)
- > - 在通用类基础上做功能重载或功能追加
- > - 注意重载时不要覆盖原始操作,避免原始提供的功能丢失
接口:
public interface BookService extends IService<Book> { // 追加的操作与原始操作通过名称区分,功能类似 boolean saveBook(Book book); boolean modify(Book book); boolean delete(Integer id); IPage<Book> getPage(int currentPage, int pageSize); IPage<Book> getPage(int currentPage, int pageSize, Book book); }
实现类:
@Service public class BookServiceImpl extends ServiceImpl<BookDao, Book> implements BookService { @Autowired private BookDao bookDao; @Override public boolean saveBook(Book book) { return bookDao.insert(book) > 0; } @Override public boolean modify(Book book) { return bookDao.updateById(book) > 0; } @Override public boolean delete(Integer id) { return bookDao.deleteById(id) > 0; } @Override public IPage<Book> getPage(int currentPage, int pageSize) { IPage page = new Page(currentPage, pageSize); bookDao.selectPage(page, null); return page; } @Override public IPage<Book> getPage(int currentPage, int pageSize, Book book) { LambdaQueryWrapper<Book> lqw = new LambdaQueryWrapper<Book>(); lqw.like(Strings.isNotEmpty(book.getType()), Book::getType, book.getType()); lqw.like(Strings.isNotEmpty(book.getName()), Book::getName, book.getName()); lqw.like(Strings.isNotEmpty(book.getDescription()), Book::getDescription, book.getDescription()); IPage page = new Page(currentPage, pageSize); bookDao.selectPage(page, lqw); return page; } }
8 基于 Restful 进行表现层开发
- 基于Restful制作表现层接口
- 新增:POST
- 删除:DELETE
- 修改:PUT
- 查询:GET
- 接收参数
- 实体数据:
@RequestBody
- 路径变量:
@PathVariable
- 实体数据:
// 功能测试 @GetMapping("/{currentPage}/{pageSize}") public R getPage(@PathVariable int currentPage, @PathVariable int pageSize, Book book) { IPage<Book> page = bookService.getPage(currentPage, pageSize, book); // 如果当前页码大于了总页码,那么将最大页码值作为当前页码,重新执行查询操作 // 源码中 long pages = this.getTotal() / this.getSize(); if (currentPage > page.getPages()) { page = bookService.getPage((int) page.getPages(), pageSize, book); } return new R(true, page); }
9 表现层消息一致性处理 R(统一返回值)
设计表现层返回结果的模型类,用于后端与前端进行数据格式统一,也称为前后端数据协议:
- 1. 设计统一的返回值结果类型便于前端开发读取数据
- 2. 返回值结果类型可以根据需求自行设定,没有固定格式
- 3. 返回值结果模型类用于后端与前端进行数据格式统一,也称为前后端数据协议
- - flag:false
- - Data: null
- - 消息(msg): 要显示信息
10 前后端协议联调
- 前后端分离结构设计中页面归属前端服务器
- 单体工程中页面放置在 resources / static 目录下(建议执行clean)
- 前端发送异步请求,调用后端接口
- created钩子函数用于初始化页面时发起调用
- 页面使用 axios 发送异步请求获取数据后确认前后端是否联通
//列表 getAll() { axios.get("/books").then((res)=>{ console.log(res.data); }); },
查询
将查询数据返回到页面,利用前端数据双向绑定进行数据展示:
//列表 getAll() { axios.get("/books").then((res)=>{ this.dataList = res.data.data; }); },
添加
- 1. 请求方式使用POST调用后台对应操作
- 2. 添加操作结束后动态刷新页面加载数据
- 3. 根据操作结果不同,显示对应的提示信息
- 4. 弹出添加Div时清除表单数据
//弹出添加窗口 handleCreate() { this.dialogFormVisible = true; }, //清除数据,重置表单 resetForm() { this.formData = {}; }, //弹出添加窗口 handleCreate() { this.dialogFormVisible = true; this.resetForm(); }, //添加 handleAdd () { //发送异步请求 axios.post("/books",this.formData).then((res)=>{ //如果操作成功,关闭弹层,显示数据 if(res.data.flag){ this.dialogFormVisible = false; this.$message.success("添加成功"); }else { this.$message.error("添加失败"); } }).finally(()=>{ this.getAll(); }); }, //取消添加 cancel(){ this.dialogFormVisible = false; this.$message.info("操作取消"); },
删除
- 1. 请求方式使用Delete调用后台对应操作
- 2. 删除操作需要传递当前行数据对应的id值到后台
- 3. 删除操作结束后动态刷新页面加载数据
- 4. 根据操作结果不同,显示对应的提示信息
- 5. 删除操作前弹出提示框避免误操作
// 删除 handleDelete(row) { axios.delete("/books/"+row.id).then((res)=>{ if(res.data.flag){ this.$message.success("删除成功"); }else{ this.$message.error("删除失败"); } }).finally(()=>{ this.getAll(); }); } // 删除 handleDelete(row) { //1.弹出提示框 this.$confirm("此操作永久删除当前数据,是否继续?","提示",{ type:'info' }).then(()=>{ //2.做删除业务 axios.delete("/books/"+row.id).then((res)=>{ …… }).finally(()=>{ this.getAll(); }); }).catch(()=>{ //3.取消删除 this.$message.info("取消删除操作"); }); }
//弹出编辑窗口 handleUpdate(row) { axios.get("/books/"+row.id).then((res)=>{ if(res.data.flag){ //展示弹层,加载数据 this.formData = res.data.data; this.dialogFormVisible4Edit = true; }else{ this.$message.error("数据同步失败,自动刷新"); } }); }, //删除 handleDelete(row) { axios.delete("/books/"+row.id).then((res)=>{ if(res.data.flag){ this.$message.success("删除成功"); }else{ this.$message.error("数据同步失败,自动刷新"); } }).finally(()=>{ this.getAll(); }); }
1.加载要修改数据通过传递当前行数据对应的id值到后台查询数据 2.利用前端数据双向绑定将查询到的数据进行回显
修改
- 1. 请求方式使用PUT调用后台对应操作
- 2. 修改操作结束后动态刷新页面加载数据(同新增)
- 3. 根据操作结果不同,显示对应的提示信息(同新增)
//修改 handleEdit() { axios.put("/books",this.formData).then((res)=>{ //如果操作成功,关闭弹层并刷新页面 if(res.data.flag){ this.dialogFormVisible4Edit = false; this.$message.success("修改成功"); }else { this.$message.error("修改失败,请重试"); } }).finally(()=>{ this.getAll(); }); }, // 取消添加和修改 cancel(){ this.dialogFormVisible = false; this.dialogFormVisible4Edit = false; this.$message.info("操作取消"); },
11 业务消息一致性处理
对异常进行统一处理,出现异常后,返回指定信息:
- 使用注解
@RestControllerAdvice
定义 SpringMVC 异常处理器用来处理异常的 - 异常处理器必须被扫描加载,否则无法生效
- 表现层返回结果的模型类中添加消息属性用来传递消息到页面
// 作为springmvc的异常处理器 @RestControllerAdvice public class ProjectExceptionAdvice { // 拦截所有的异常信息 @ExceptionHandler(Exception.class) public R doException(Exception e){ // 记录日志 // 通知运维 // 通知开发 e.printStackTrace(); return new R("服务器故障,请稍后重试哈!"); } }
12 分页功能
页面使用 el 分页组件添加分页功能:
- 定义分页组件需要使用的数据并将数据绑定到分页组件
- 替换查询全部功能为分页功能
- 加载分页数据
- 分页页码值切换
使用el分页组件:
- 定义分页组件绑定的数据模型
- 异步调用获取分页数据
- 分页数据页面回显
到此这篇关于SpringBoot SSMP 整合案例分享的文章就介绍到这了,更多相关SpringBoot SSMP 内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!