SpringBoot SSMP

软件发布|下载排行|最新软件

当前位置:首页IT学院IT技术

SpringBoot SSMP

Lemonade22​​​​​​​   2022-09-28 我要评论

前言:

  • - 先开发基础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分页组件:

  • 定义分页组件绑定的数据模型
  • 异步调用获取分页数据
  • 分页数据页面回显

Copyright 2022 版权所有 软件发布 访问手机版

声明:所有软件和文章来自软件开发商或者作者 如有异议 请与本站联系 联系我们