• 二十三、SpringBoot + Jwt + Vue 权限管理系统 (4)


    后端开发开始了哦 🐄🐄

    一、项目后端搭建

    1.1 前言

    • 从零开始搭建一个项目骨架,最好选择合适熟悉的技术,并且在未来易拓展,适合微服务化体系等。所以一般以Springboot作为我们的框架基础,这是离不开的了。
    • 然后数据层,我们常用的是Mybatis,易上手,方便维护。但是单表操作比较困难,特别是添加字段或减少字段的时候,比较繁琐,所以这里我推荐使用Mybatis Plus(https://mp.baomidou.com/),为简化开发而生,只需简单配置,即可快速进行 CRUD 操作,从而节省大量时间。
    • 作为一个项目骨架,权限也是我们不能忽略的,所以我们使用security作为我们的权限控制和会话控制的框架。
    • 考虑到项目可能需要部署多台,一些需要共享的信息就保存在中间件中,Redis是现在主流的缓存中间件,也适合我们的项目。
    • 然后因为前后端分离,所以我们使用jwt作为我们用户身份凭证,并且session我们会禁用,这样以前传统项目使用的方式我们可能就不再适合使用,这点需要注意了。

    ok,我们现在就开始搭建我们的项目脚手架

    技术栈:

    • SpringBoot
    • mybatis plus
    • spring security
    • lombok
    • redis
    • hibernate validatior
    • jwt

    1.2 新建springboot项目,注意版本

    这里,我们使用IDEA来开发我们项目,新建步骤比较简单。

    开发工具与环境:

    • idea
    • mysql
    • jdk 8
    • maven3.3.9

    项目创建步骤如下喽 👇🏾👇🏾,SpringBoot版本目前使用的是2.4.0版本

    (1)
    在这里插入图片描述

    (2)

    • devtools:项目的热加载重启插件
    • lombok:简化代码的工具

    在这里插入图片描述

    (3)

    在这里插入图片描述

    (4)删掉自动生成没用的

    在这里插入图片描述

    (5) 2.4.0版本

    在这里插入图片描述

    二、整合mybatis plus,生成代码

    • 接下来,整合mybatis plus,让项目能完成基本的增删改查操作。
    • 步骤很简单:也可以去官网看看 MyBatis-Plus

    2.1 导入jar包

    pom中导入mybatis plus的jar包,因为后面会涉及到代码生成,所以我们还需要导入页面模板引擎,这里我们用的是freemarker。

    在这里插入图片描述

    
    <dependency>
        <groupId>com.baomidougroupId>
        <artifactId>mybatis-plus-boot-starterartifactId>
        <version>3.4.1version>
    dependency>
    <dependency>
        <groupId>com.baomidougroupId>
        <artifactId>mybatis-plus-generatorartifactId>
        <version>3.4.1version>
    dependency>
    <dependency>
        <groupId>org.freemarkergroupId>
        <artifactId>freemarkerartifactId>
        <version>2.3.30version>
    dependency>
    <dependency>
        <groupId>mysqlgroupId>
        <artifactId>mysql-connector-javaartifactId>
        <version>5.1.32version>
    dependency>
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    2.2 去写配置文件

    上面除了配置数据库的信息,还配置了myabtis plus的mapper的xml文件的扫描路径,这一步不要忘记了。然后因为前段默认是8080端口了,所以后端我们设置为8081端口,防止端口冲突。

    在这里插入图片描述

    application.yml

    server:
      port: 8081
    # DataSource Config
    spring:
      datasource:
           driver-class-name: com.mysql.jdbc.Driver
           url: jdbc:mysql://localhost:3306/daniel-admin?useUnicode=true&useSSL=false&characterEncoding=utf8&serverTimezone=Asia/Shanghai
           username: root
           password: root
    mybatis-plus:
      mapper-locations: classpath*:/mapper/**Mapper.xml
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    2.3 开启mapper接口扫描,添加分页、防全表更新插件

    新建一个包(com.zql.config):通过@mapperScan注解指定要变成实现类的接口所在的包,然后包下面的所有接口在编译之后都会生成相应的实现类。

    com.zql.config.MybatisPlusConfig

    MybatisPlusConfig.java

    参考官网

    在这里插入图片描述

    在这里插入图片描述

    package com.zql.config;
    
    import com.baomidou.mybatisplus.annotation.DbType;
    import com.baomidou.mybatisplus.autoconfigure.ConfigurationCustomizer;
    import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
    import com.baomidou.mybatisplus.extension.plugins.inner.BlockAttackInnerInterceptor;
    import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
    import org.mybatis.spring.annotation.MapperScan;
    import org.springframework.context.annotation.Bean;
    import org.springframework.context.annotation.Configuration;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    @Configuration
    @MapperScan("com.zql.mapper")
    public class MybatisPlusConfig {
    
        /* 新的分页插件,一缓和二缓遵循mybatis的规则,
         *  需要设置 MybatisConfiguration#useDeprecatedExecutor = false
         *  避免缓存出现问题(该属性会在旧插件移除后一同移除)
         */
        @Bean
        public MybatisPlusInterceptor mybatisPlusInterceptor() {
            MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
            interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));
            // 防止全表更新和删除
            interceptor.addInnerInterceptor(new BlockAttackInnerInterceptor());
            return interceptor;
        }
        @Bean
        public ConfigurationCustomizer configurationCustomizer() {
            return configuration -> configuration.setUseDeprecatedExecutor(false);
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36

    上面代码中,我们给Mybatis plus添加了2个拦截器,这是根据 mp官网配置的上面截图已显示:

    PaginationInnerInterceptor:新的分页插件
    BlockAttackInnerInterceptor:防止全表更新和删除

    2.4 创建数据库和表

    因为是后台管理系统的权限模块,所以我们需要考虑的表主要就几个:用户表(sys_user)角色表(sys_role)菜单权限表(sys_menu)、以及关联的用户角色中间表(sys_user_role)菜单角色中间表(sys_role_menu)。就 5 个表,至于什么字段其实都很随意的,用户表里面除了用户名、密码字段必要,其他其实都随意,然后角色和菜单我们可以参考一下其他的系统、或者自己在做项目的过程中需要的时候在添加也行,反正重新生成代码也是非常简便的事情,综合考虑,数据库名称为 daniel-admin ,建表语句如下:

    在这里插入图片描述

    在这里插入图片描述

    用户名/密码 admin/111111 test/1234567

    SET FOREIGN_KEY_CHECKS=0;
    
    -- ----------------------------
    -- Table structure for sys_menu
    -- ----------------------------
    DROP TABLE IF EXISTS `sys_menu`;
    CREATE TABLE `sys_menu` (
      `id` BIGINT(20) NOT NULL AUTO_INCREMENT,
      `parent_id` BIGINT(20) DEFAULT NULL COMMENT '父菜单ID,一级菜单为0',
      `name` VARCHAR(64) NOT NULL,
      `path` VARCHAR(255) DEFAULT NULL COMMENT '菜单URL',
      `perms` VARCHAR(255) DEFAULT NULL COMMENT '授权(多个用逗号分隔,如:user:list,user:create)',
      `component` VARCHAR(255) DEFAULT NULL,
      `type` INT(5) NOT NULL COMMENT '类型     0:目录   1:菜单   2:按钮',
      `icon` VARCHAR(32) DEFAULT NULL COMMENT '菜单图标',
      `orderNum` INT(11) DEFAULT NULL COMMENT '排序',
      `created` DATETIME NOT NULL,
      `updated` DATETIME DEFAULT NULL,
      `statu` INT(5) NOT NULL,
      PRIMARY KEY (`id`),
      UNIQUE KEY `name` (`name`) USING BTREE
    ) ENGINE=INNODB AUTO_INCREMENT=20 DEFAULT CHARSET=utf8;
    
    -- ----------------------------
    -- Records of sys_menu
    -- ----------------------------
    INSERT INTO `sys_menu` VALUES ('1', '0', '系统管理', '', 'sys:manage', '', '0', 'el-icon-s-operation', '1', '2022-01-15 18:58:18', '2022-01-15 18:58:20', '1');
    INSERT INTO `sys_menu` VALUES ('2', '1', '用户管理', '/sys/users', 'sys:user:list', 'sys/User', '1', 'el-icon-s-custom', '1', '2022-01-15 19:03:45', '2022-01-15 19:03:48', '1');
    INSERT INTO `sys_menu` VALUES ('3', '1', '角色管理', '/sys/roles', 'sys:role:list', 'sys/Role', '1', 'el-icon-rank', '2', '2022-01-15 19:03:45', '2022-01-15 19:03:48', '1');
    INSERT INTO `sys_menu` VALUES ('4', '1', '菜单管理', '/sys/menus', 'sys:menu:list', 'sys/Menu', '1', 'el-icon-menu', '3', '2022-01-15 19:03:45', '2022-01-15 19:03:48', '1');
    INSERT INTO `sys_menu` VALUES ('5', '0', '系统工具', '', 'sys:tools', NULL, '0', 'el-icon-s-tools', '2', '2022-01-15 19:06:11', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('6', '5', '数字字典', '/sys/dicts', 'sys:dict:list', 'sys/Dict', '1', 'el-icon-s-order', '1', '2022-01-15 19:07:18', '2022-01-18 16:32:13', '1');
    INSERT INTO `sys_menu` VALUES ('7', '3', '添加角色', '', 'sys:role:save', '', '2', '', '1', '2022-01-15 23:02:25', '2022-01-17 21:53:14', '0');
    INSERT INTO `sys_menu` VALUES ('9', '2', '添加用户', NULL, 'sys:user:save', NULL, '2', NULL, '1', '2022-01-17 21:48:32', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('10', '2', '修改用户', NULL, 'sys:user:update', NULL, '2', NULL, '2', '2022-01-17 21:49:03', '2022-01-17 21:53:04', '1');
    INSERT INTO `sys_menu` VALUES ('11', '2', '删除用户', NULL, 'sys:user:delete', NULL, '2', NULL, '3', '2022-01-17 21:49:21', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('12', '2', '分配角色', NULL, 'sys:user:role', NULL, '2', NULL, '4', '2022-01-17 21:49:58', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('13', '2', '重置密码', NULL, 'sys:user:repass', NULL, '2', NULL, '5', '2022-01-17 21:50:36', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('14', '3', '修改角色', NULL, 'sys:role:update', NULL, '2', NULL, '2', '2022-01-17 21:51:14', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('15', '3', '删除角色', NULL, 'sys:role:delete', NULL, '2', NULL, '3', '2022-01-17 21:51:39', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('16', '3', '分配权限', NULL, 'sys:role:perm', NULL, '2', NULL, '5', '2022-01-17 21:52:02', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('17', '4', '添加菜单', NULL, 'sys:menu:save', NULL, '2', NULL, '1', '2022-01-17 21:53:53', '2022-01-17 21:55:28', '1');
    INSERT INTO `sys_menu` VALUES ('18', '4', '修改菜单', NULL, 'sys:menu:update', NULL, '2', NULL, '2', '2022-01-17 21:56:12', NULL, '1');
    INSERT INTO `sys_menu` VALUES ('19', '4', '删除菜单', NULL, 'sys:menu:delete', NULL, '2', NULL, '3', '2022-01-17 21:56:36', NULL, '1');
    
    -- ----------------------------
    -- Table structure for sys_role
    -- ----------------------------
    DROP TABLE IF EXISTS `sys_role`;
    CREATE TABLE `sys_role` (
      `id` BIGINT(20) NOT NULL AUTO_INCREMENT,
      `name` VARCHAR(64) NOT NULL,
      `code` VARCHAR(64) NOT NULL,
      `remark` VARCHAR(64) DEFAULT NULL COMMENT '备注',
      `created` DATETIME DEFAULT NULL,
      `updated` DATETIME DEFAULT NULL,
      `statu` INT(5) NOT NULL,
      PRIMARY KEY (`id`),
      UNIQUE KEY `name` (`name`) USING BTREE,
      UNIQUE KEY `code` (`code`) USING BTREE
    ) ENGINE=INNODB AUTO_INCREMENT=7 DEFAULT CHARSET=utf8;
    
    -- ----------------------------
    -- Records of sys_role
    -- ----------------------------
    INSERT INTO `sys_role` VALUES ('3', '普通用户', 'normal', '只有基本查看功能', '2022-01-04 10:09:14', '2022-01-30 08:19:52', '1');
    INSERT INTO `sys_role` VALUES ('6', '超级管理员', 'admin', '系统默认最高权限,不可以编辑和任意修改', '2022-01-16 13:29:03', '2022-01-17 15:50:45', '1');
    
    -- ----------------------------
    -- Table structure for sys_role_menu
    -- ----------------------------
    DROP TABLE IF EXISTS `sys_role_menu`;
    CREATE TABLE `sys_role_menu` (
      `id` BIGINT(20) NOT NULL AUTO_INCREMENT,
      `role_id` BIGINT(20) NOT NULL,
      `menu_id` BIGINT(20) NOT NULL,
      PRIMARY KEY (`id`)
    ) ENGINE=INNODB AUTO_INCREMENT=102 DEFAULT CHARSET=utf8mb4;
    
    -- ----------------------------
    -- Records of sys_role_menu
    -- ----------------------------
    INSERT INTO `sys_role_menu` VALUES ('60', '6', '1');
    INSERT INTO `sys_role_menu` VALUES ('61', '6', '2');
    INSERT INTO `sys_role_menu` VALUES ('62', '6', '9');
    INSERT INTO `sys_role_menu` VALUES ('63', '6', '10');
    INSERT INTO `sys_role_menu` VALUES ('64', '6', '11');
    INSERT INTO `sys_role_menu` VALUES ('65', '6', '12');
    INSERT INTO `sys_role_menu` VALUES ('66', '6', '13');
    INSERT INTO `sys_role_menu` VALUES ('67', '6', '3');
    INSERT INTO `sys_role_menu` VALUES ('68', '6', '7');
    INSERT INTO `sys_role_menu` VALUES ('69', '6', '14');
    INSERT INTO `sys_role_menu` VALUES ('70', '6', '15');
    INSERT INTO `sys_role_menu` VALUES ('71', '6', '16');
    INSERT INTO `sys_role_menu` VALUES ('72', '6', '4');
    INSERT INTO `sys_role_menu` VALUES ('73', '6', '17');
    INSERT INTO `sys_role_menu` VALUES ('74', '6', '18');
    INSERT INTO `sys_role_menu` VALUES ('75', '6', '19');
    INSERT INTO `sys_role_menu` VALUES ('76', '6', '5');
    INSERT INTO `sys_role_menu` VALUES ('77', '6', '6');
    INSERT INTO `sys_role_menu` VALUES ('96', '3', '1');
    INSERT INTO `sys_role_menu` VALUES ('97', '3', '2');
    INSERT INTO `sys_role_menu` VALUES ('98', '3', '3');
    INSERT INTO `sys_role_menu` VALUES ('99', '3', '4');
    INSERT INTO `sys_role_menu` VALUES ('100', '3', '5');
    INSERT INTO `sys_role_menu` VALUES ('101', '3', '6');
    
    -- ----------------------------
    -- Table structure for sys_user
    -- ----------------------------
    DROP TABLE IF EXISTS `sys_user`;
    CREATE TABLE `sys_user` (
      `id` BIGINT(20) NOT NULL AUTO_INCREMENT,
      `username` VARCHAR(64) DEFAULT NULL,
      `password` VARCHAR(64) DEFAULT NULL,
      `avatar` VARCHAR(255) DEFAULT NULL,
      `email` VARCHAR(64) DEFAULT NULL,
      `city` VARCHAR(64) DEFAULT NULL,
      `created` DATETIME DEFAULT NULL,
      `updated` DATETIME DEFAULT NULL,
      `last_login` DATETIME DEFAULT NULL,
      `statu` INT(5) NOT NULL,
      PRIMARY KEY (`id`),
      UNIQUE KEY `UK_USERNAME` (`username`) USING BTREE
    ) ENGINE=INNODB AUTO_INCREMENT=3 DEFAULT CHARSET=utf8;
    
    -- ----------------------------
    -- Records of sys_user
    -- ----------------------------
    INSERT INTO `sys_user` VALUES ('1', 'admin', '$2a$10$R7zegeWzOXPw871CmNuJ6upC0v8D373GuLuTw8jn6NET4BkPRZfgK', 'https://image-1300566513.cos.ap-guangzhou.myqcloud.com/upload/images/5a9f48118166308daba8b6da7e466aab.jpg', '123@qq.com', '广州', '2021-01-12 22:13:53', '2021-01-16 16:57:32', '2020-12-30 08:38:37', '1');
    INSERT INTO `sys_user` VALUES ('2', 'test', '$2a$10$0ilP4ZD1kLugYwLCs4pmb.ZT9cFqzOZTNaMiHxrBnVIQUGUwEvBIO', 'https://image-1300566513.cos.ap-guangzhou.myqcloud.com/upload/images/5a9f48118166308daba8b6da7e466aab.jpg', 'test@qq.com', NULL, '2021-01-30 08:20:22', '2021-01-30 08:55:57', NULL, '1');
    
    -- ----------------------------
    -- Table structure for sys_user_role
    -- ----------------------------
    DROP TABLE IF EXISTS `sys_user_role`;
    CREATE TABLE `sys_user_role` (
      `id` BIGINT(20) NOT NULL AUTO_INCREMENT,
      `user_id` BIGINT(20) NOT NULL,
      `role_id` BIGINT(20) NOT NULL,
      PRIMARY KEY (`id`)
    ) ENGINE=INNODB AUTO_INCREMENT=14 DEFAULT CHARSET=utf8mb4;
    
    -- ----------------------------
    -- Records of sys_user_role
    -- ----------------------------
    INSERT INTO `sys_user_role` VALUES ('4', '1', '6');
    INSERT INTO `sys_user_role` VALUES ('7', '1', '3');
    INSERT INTO `sys_user_role` VALUES ('13', '2', '3');
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144
    • 145
    • 146
    • 147
    • 148
    • 149

    2.5 代码生成

    1. 获取项目数据库所对应表和字段的信息
    2. 新建一个freemarker的页面模板 - SysUser.java.ftl - ${baseEntity}
    3. 提供相关需要进行渲染的动态数据 - BaseEntity、表字段、注释、baseEntity=SuperEntity
    4. 使用freemarker模板引擎进行渲染! - SysUser.java
    # 获取表
    SELECT	*
    FROM	
    information_schema. TABLES
    WHERE	
    TABLE_SCHEMA = (SELECT DATABASE());
    
    
    # 获取字段
    SELECT	*
    FROM	
    information_schema. COLUMNS
    WHERE	
    TABLE_SCHEMA = (SELECT DATABASE())
    AND TABLE_NAME = "sys_user";
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15

    在这里插入图片描述

    (1)

    有了数据库之后,那么现在就已经可以使用mybatis plus了,官方给我们提供了一个代码生成器,然后我写上自己的参数之后,就可以直接根据数据库表信息生成entityservicemapper等接口和实现类。

    com.zql.CodeGenerator

    官网 MyBatis-Plus代码生成器

    在这里插入图片描述

    (2) CodeGenerator.java

    在这里插入图片描述

    package com.zql;
    
    import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
    import com.baomidou.mybatisplus.core.toolkit.StringPool;
    import com.baomidou.mybatisplus.core.toolkit.StringUtils;
    import com.baomidou.mybatisplus.generator.AutoGenerator;
    import com.baomidou.mybatisplus.generator.InjectionConfig;
    import com.baomidou.mybatisplus.generator.config.*;
    import com.baomidou.mybatisplus.generator.config.po.TableInfo;
    import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
    import com.baomidou.mybatisplus.generator.engine.FreemarkerTemplateEngine;
    
    import java.util.ArrayList;
    import java.util.List;
    import java.util.Scanner;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    public class CodeGenerator {
    
        /**
         * 

    * 读取控制台内容 *

    */
    public static String scanner(String tip) { Scanner scanner = new Scanner(System.in); StringBuilder help = new StringBuilder(); help.append("请输入" + tip + ":"); System.out.println(help.toString()); if (scanner.hasNext()) { String ipt = scanner.next(); if (StringUtils.isNotBlank(ipt)) { return ipt; } } throw new MybatisPlusException("请输入正确的" + tip + "!"); } public static void main(String[] args) { // 代码生成器 AutoGenerator mpg = new AutoGenerator(); // 全局配置 GlobalConfig gc = new GlobalConfig(); String projectPath = System.getProperty("user.dir"); gc.setOutputDir(projectPath + "/src/main/java"); gc.setAuthor("我的公众号:爪哇知识库"); gc.setOpen(false); // gc.setSwagger2(true); 实体属性 Swagger2 注解 gc.setServiceName("%sService"); mpg.setGlobalConfig(gc); // 数据源配置 DataSourceConfig dsc = new DataSourceConfig(); dsc.setUrl("jdbc:mysql://localhost:3306/daniel-admin?useUnicode=true&useSSL=false&characterEncoding=utf8"); // dsc.setSchemaName("public"); dsc.setDriverName("com.mysql.jdbc.Driver"); dsc.setUsername("root"); dsc.setPassword("root"); mpg.setDataSource(dsc); // 包配置 PackageConfig pc = new PackageConfig(); // pc.setModuleName(scanner("模块名")); pc.setParent("com.zql"); mpg.setPackageInfo(pc); // 自定义配置 InjectionConfig cfg = new InjectionConfig() { @Override public void initMap() { // to do nothing } }; // 如果模板引擎是 freemarker String templatePath = "/templates/mapper.xml.ftl"; // 如果模板引擎是 velocity // String templatePath = "/templates/mapper.xml.vm"; // 自定义输出配置 List<FileOutConfig> focList = new ArrayList<>(); // 自定义配置会被优先输出 focList.add(new FileOutConfig(templatePath) { @Override public String outputFile(TableInfo tableInfo) { // 自定义输出文件名 , 如果你 Entity 设置了前后缀、此处注意 xml 的名称会跟着发生变化!! return projectPath + "/src/main/resources/mapper/" + pc.getModuleName() + "/" + tableInfo.getEntityName() + "Mapper" + StringPool.DOT_XML; } }); /* cfg.setFileCreate(new IFileCreate() { @Override public boolean isCreate(ConfigBuilder configBuilder, FileType fileType, String filePath) { // 判断自定义文件夹是否需要创建 checkDir("调用默认方法创建的目录,自定义目录用"); if (fileType == FileType.MAPPER) { // 已经生成 mapper 文件判断存在,不想重新生成返回 false return !new File(filePath).exists(); } // 允许生成模板文件 return true; } }); */ cfg.setFileOutConfigList(focList); mpg.setCfg(cfg); // 配置模板 TemplateConfig templateConfig = new TemplateConfig(); // 配置自定义输出模板 //指定自定义模板路径,注意不要带上.ftl/.vm, 会根据使用的模板引擎自动识别 // templateConfig.setEntity("templates/entity2.java"); // templateConfig.setService(); // templateConfig.setController(); templateConfig.setXml(null); mpg.setTemplate(templateConfig); // 策略配置 StrategyConfig strategy = new StrategyConfig(); strategy.setNaming(NamingStrategy.underline_to_camel); strategy.setColumnNaming(NamingStrategy.underline_to_camel); strategy.setSuperEntityClass("BaseEntity"); strategy.setEntityLombokModel(true); strategy.setRestControllerStyle(true); // 公共父类 strategy.setSuperControllerClass("BaseController"); // 写于父类中的公共字段 strategy.setSuperEntityColumns("id","created","updated","statu"); strategy.setInclude(scanner("表名,多个英文逗号分割").split(",")); strategy.setControllerMappingHyphenStyle(true); strategy.setTablePrefix(pc.getModuleName() + "_"); mpg.setStrategy(strategy); mpg.setTemplateEngine(new FreemarkerTemplateEngine()); mpg.execute(); } }
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
    • 85
    • 86
    • 87
    • 88
    • 89
    • 90
    • 91
    • 92
    • 93
    • 94
    • 95
    • 96
    • 97
    • 98
    • 99
    • 100
    • 101
    • 102
    • 103
    • 104
    • 105
    • 106
    • 107
    • 108
    • 109
    • 110
    • 111
    • 112
    • 113
    • 114
    • 115
    • 116
    • 117
    • 118
    • 119
    • 120
    • 121
    • 122
    • 123
    • 124
    • 125
    • 126
    • 127
    • 128
    • 129
    • 130
    • 131
    • 132
    • 133
    • 134
    • 135
    • 136
    • 137
    • 138
    • 139
    • 140
    • 141
    • 142
    • 143
    • 144

    (3) 运行生成, 控制台输入表名: sys_user,sys_role,sys_menu,sys_role_menu,sys_user_role

    在这里插入图片描述

    (4) 使用快捷键 Ctrl+Shift+f 删除多余的包 import.BaseController \ import.BaseEntity

    在这里插入图片描述
    在这里插入图片描述

    (5) 如果mapper没有显示,且生成了,则刷新即可出现

    在这里插入图片描述

    (6) 然后创建 BaseController 和 BaseEntity

    生成和创建如下图所示

    在这里插入图片描述在这里插入图片描述

    (7)

    上面代码生成的过程中,我默认所有的实体类都继承BaseEntity,控制器都继承BaseController,所以在代码生成之前,最好先编写这两个基类:

    • com.zql.entity.BaseEntity
    • com.zql.controller.BaseController

    BaseController.java

    package com.zql.controller;
    
    import org.springframework.beans.factory.annotation.Autowired;
    
    import javax.servlet.http.HttpServletRequest;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    public class BaseController {   
        @Autowired   
        HttpServletRequest req;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    (8)

    这里有点需要注意,因为关联的用户角色中间表、菜单角色中间表我们是没有created等几个公共字段的,所以我们把这两个实体继承BaseEntity去掉👇🏾👇🏾:

    在这里插入图片描述

    package com.zql.entity;
    
    import lombok.Data;
    import lombok.EqualsAndHashCode;
    
    /**
     * 

    * *

    * * @author Daniel * @since 2022-11-05 */
    @Data //@EqualsAndHashCode(callSuper = true) public class SysRoleMenu{ private static final long serialVersionUID = 1L; private Long roleId; private Long menuId; }
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

    2.6 测试生成的代码

    简洁!方便!经过上面的步骤,基本上我们已经把mybatis plus框架集成到项目中了,并且也生成了基本的代码,省了好多功夫。然后我们做个简单测试:

    com.zql.controller.TestController

    package com.zql.controller;
    
    import com.zql.service.SysUserService;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RequestBody;
    import org.springframework.web.bind.annotation.ResponseBody;
    import org.springframework.web.bind.annotation.RestController;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    
    @RestController
    public class TestController {
    
        @Autowired
        SysUserService sysUserService;
    
        @GetMapping("/test")
        public Object test(){
    
            return  sysUserService.list();
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    测试显示 : http://localhost:8081/test

    在这里插入图片描述

    调整浏览器显示如下,可在浏览器装个JsonView插件即可;

    在这里插入图片描述
    没有显示 id是因为公共部分需要我们手动去写

    • com.zql.entity.BaseEntity
    package com.zql.entity;
    
    import com.baomidou.mybatisplus.annotation.IdType;
    import com.baomidou.mybatisplus.annotation.TableId;
    import lombok.Data;
    
    import java.io.Serializable;
    import java.time.LocalDateTime;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    @Data
    public class BaseEntity implements Serializable {
    
        @TableId(value = "id", type = IdType.AUTO)
        private Long id;
        private LocalDateTime created;
        private LocalDateTime updated;
        private Integer statu;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    再次测试得:http://localhost:8081/test

    在这里插入图片描述

    三、结果封装

    在这里插入图片描述

    详细看下面 👇🏾👇🏾

    因为是前后端分离的项目,所以我们有必要统一 一个结果返回封装类,这样前后端交互的时候有个统一的标准,约定结果返回的数据是正常的或者遇到异常了。

    这里我们用到了一个Result的类,这个用于我们的异步统一返回的结果封装。一般来说,结果里面有几个要素必要的

    • 是否成功,可用code表示(如200表示成功,400表示异常)
    • 结果消息
    • 结果数据

    所以可得到封装如下:

    1. 创建包 com.zql.common.lang 再创建封装类 Result.java

    Result.java

    package com.zql.common.lang;
    
    import lombok.Data;
    
    import java.io.Serializable;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    @Data
    public class Result implements Serializable {
    
        private int code;
        private String msg;
        private Object data;
    
        public static Result succ(Object data){
            return succ(200,"操作成功",data);
        }
    
        public static Result succ(int code,String msg,Object data){
    
            Result r = new Result();
    
            r.setCode(code);
            r.setMsg(msg);
            r.setData(data);
            return  r;
        }
    
        public static Result fail(String msg,Object data){
            return succ(400,msg,null);
        }
        public static Result fail(int code,String msg,Object data){
    
            Result r = new Result();
            r.setCode(code);
            r.setMsg(msg);
            r.setData(data);
            return  r;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    1. 测试:修改 TestController.java

    在这里插入图片描述

    package com.zql.controller;
    
    import com.zql.common.lang.Result;
    import com.zql.service.SysUserService;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.stereotype.Controller;
    import org.springframework.web.bind.annotation.GetMapping;
    import org.springframework.web.bind.annotation.RequestBody;
    import org.springframework.web.bind.annotation.ResponseBody;
    import org.springframework.web.bind.annotation.RestController;
    
    /**
     * @Author:Daniel
     * @Version 1.0
     */
    
    @RestController
    public class TestController {
    
        @Autowired
        SysUserService sysUserService;
    
        @GetMapping("/test")
        public Result test(){
    
            return  Result.succ(sysUserService.list());
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    1. 运行主程序启动可以看如下所示:http://localhost:8081/test

    在这里插入图片描述

    : 除了在结果封装类上的code可以体现数据是否正常,我们还可以通过http的状态码来体现访问是否遇到了异常,比如401表示五权限拒绝访问等,注意灵活使用。

    四、全局异常处理

    有时候不可避免服务器报错的情况,如果不配置异常处理机制,就会默认返回tomcat或者nginx的5XX页面,对普通用户来说,不太友好,用户也不懂什么情况。这时候需要我们程序员设计返回一个友好简单的格式给前端。

    处理办法如下:通过使用@ControllerAdvice来进行统一异常处理,@ExceptionHandler(value = RuntimeException.class)来指定捕获的Exception各个类型异常 ,这个异常的处理,是全局的,所有类似的异常,都会跑到这个地方处理。

    步骤二、定义全局异常处理,@ControllerAdvice表示定义全局控制器异常处理,@ExceptionHandler表示针对性异常处理,可对每种异常针对性处理。

    1. 创建,而后直接copy就可以 com.zql.common.exception.GlobalExceptionHandler

    GlobalExceptionHandler.java

    package com.zql.common.exception;
    
    import com.zql.common.lang.Result;
    import lombok.extern.slf4j.Slf4j;
    import org.springframework.http.HttpStatus;
    import org.springframework.validation.BindingResult;
    import org.springframework.validation.ObjectError;
    import org.springframework.web.bind.MethodArgumentNotValidException;
    import org.springframework.web.bind.annotation.ExceptionHandler;
    import org.springframework.web.bind.annotation.ResponseStatus;
    import org.springframework.web.bind.annotation.RestControllerAdvice;
    
    import java.nio.file.AccessDeniedException;
    
    
    /**
     * @Author:Daniel
     * @Version 1.0
     * 全局异常处理
     */
    
    @Slf4j
    @RestControllerAdvice
    class GlobalExceptionHandler {
      @ResponseStatus(HttpStatus.FORBIDDEN)
        @ExceptionHandler(value = AccessDeniedException.class)
        public Result handler(AccessDeniedException e) {
            log.info("security权限不足:----------------{}", e.getMessage());
            return Result.fail("权限不足");
        }
    
        @ResponseStatus(HttpStatus.BAD_REQUEST)
        @ExceptionHandler(value = MethodArgumentNotValidException.class)
        public Result handler(MethodArgumentNotValidException e) {
            log.info("实体校验异常:----------------{}", e.getMessage());
            BindingResult bindingResult = e.getBindingResult();
            ObjectError objectError = bindingResult.getAllErrors().stream().findFirst().get();
            return Result.fail(objectError.getDefaultMessage());
        }
    
        @ResponseStatus(HttpStatus.BAD_REQUEST)
        @ExceptionHandler(value = IllegalArgumentException.class)
        public Result handler(IllegalArgumentException e) {
            log.error("Assert异常:----------------{}", e.getMessage());
            return Result.fail(e.getMessage());
        }
    
        @ResponseStatus(HttpStatus.BAD_REQUEST)
        @ExceptionHandler(value = RuntimeException.class)
        public Result handler(RuntimeException e) {
        log.error("运行时异常:----------------{}", e);
        return Result.fail(e.getMessage());
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    1. 上面我们捕捉了几个异常:(权限和实体校验也可以放在后面遇到的时候再写也没啥咯🤣🤣)
    • ShiroExceptionshiro抛出的异常,比如没有权限,用户登录异常
    • IllegalArgumentException处理Assert的异常
    • MethodArgumentNotValidException处理实体校验的异常
    • RuntimeException捕捉其他异常

    当前后端源码即文章所有源码

  • 相关阅读:
    maven清理本地仓库。删除_remote.repositories文件和删除失败的jar包
    web 面试高频考点 —— HTTP 篇
    MySQL权限
    记录一下:基于nginx配置的封禁真实IP
    构建、运行、增长,亚马逊云科技发布游戏行业解决方案
    java8函数式编程之Stream流处理的方法和案例讲解
    剑指Offer07.重建二叉树_解题思路&代码实现
    路由 —— 源站路由 + 策略路由
    Linux应用程序崩溃了怎么定位?
    Network 之十 BIOS + MBR、UEFI + GPT、GRUB、BOOTMGR、SYSLINUX、Option ROM
  • 原文地址:https://blog.csdn.net/weixin_42171159/article/details/127676019