• 三级分类的数据表设计和构造API数据


    诸如下图这种:

    一级菜单下有二级菜单和三级菜单。与此类似的应用场景还有很多很多,如省市县的三级联动、后台管理菜单、部门展示等等。

    一、数据表设计

    先看看表结构吧

    具体的 sql 文件,在贴的源码仓库中有。

    其中最重要的就是理清父子层级的关系就好了,知道是个什么样子设计的,如何去管理他们,那这些就都不是事啦。

    二级评论的表的设计方法其实也大致如此。

    数据表中对应的数据关系大致如下:

    二、项目代码

    技术就是常用的 SpringBoot、Mybatis-Plus

    另外还用了下Mybatis-Plus逆向生成器,还有Java的函数式编程

    3.1、导入依赖

    完整依赖如下:

    1. <parent>
    2. <groupId>org.springframework.bootgroupId>
    3. <artifactId>spring-boot-dependenciesartifactId>
    4. <version>2.5.2version>
    5. parent>
    6. <properties>
    7. <maven.compiler.source>8maven.compiler.source>
    8. <maven.compiler.target>8maven.compiler.target>
    9. properties>
    10. <dependencies>
    11. <dependency>
    12. <groupId>org.springframework.bootgroupId>
    13. <artifactId>spring-boot-starterartifactId>
    14. dependency>
    15. <dependency>
    16. <groupId>org.springframework.bootgroupId>
    17. <artifactId>spring-boot-starter-webartifactId>
    18. dependency>
    19. <dependency>
    20. <groupId>org.springframework.bootgroupId>
    21. <artifactId>spring-boot-starter-testartifactId>
    22. dependency>
    23. <dependency>
    24. <groupId>com.baomidougroupId>
    25. <artifactId>mybatis-plus-boot-starterartifactId>
    26. <version>3.4.1version>
    27. dependency>
    28. <dependency>
    29. <groupId>com.alibabagroupId>
    30. <artifactId>druid-spring-boot-starterartifactId>
    31. <version>1.2.6version>
    32. dependency>
    33. <dependency>
    34. <groupId>mysqlgroupId>
    35. <artifactId>mysql-connector-javaartifactId>
    36. dependency>
    37. <dependency>
    38. <groupId>org.springframework.bootgroupId>
    39. <artifactId>spring-boot-configuration-processorartifactId>
    40. <optional>trueoptional>
    41. dependency>
    42. <dependency>
    43. <groupId>com.baomidougroupId>
    44. <artifactId>mybatis-plus-generatorartifactId>
    45. <version>3.4.1version>
    46. dependency>
    47. <dependency>
    48. <groupId>org.apache.velocitygroupId>
    49. <artifactId>velocity-engine-coreartifactId>
    50. <version>2.2version>
    51. dependency>
    52. <dependency>
    53. <groupId>org.projectlombokgroupId>
    54. <artifactId>lombokartifactId>
    55. dependency>
    56. <dependency>
    57. <groupId>junitgroupId>
    58. <artifactId>junitartifactId>
    59. <version>4.12version>
    60. dependency>
    61. dependencies>

    关于 Mybatis-plus 的版本,最新的那个逆向生成器不太熟,就用了旧版本的。感兴趣的可以去试一试新版本的。

    3.2、逆向生成代码

    1. import com.baomidou.mybatisplus.core.exceptions.MybatisPlusException;
    2. import com.baomidou.mybatisplus.core.toolkit.StringPool;
    3. import com.baomidou.mybatisplus.generator.AutoGenerator;
    4. import com.baomidou.mybatisplus.generator.InjectionConfig;
    5. import com.baomidou.mybatisplus.generator.config.*;
    6. import com.baomidou.mybatisplus.generator.config.po.TableInfo;
    7. import com.baomidou.mybatisplus.generator.config.rules.NamingStrategy;
    8. import com.baomidou.mybatisplus.generator.engine.VelocityTemplateEngine;
    9. import org.junit.platform.commons.util.StringUtils;
    10. import java.util.ArrayList;
    11. import java.util.List;
    12. import java.util.Scanner;
    13. // 演示例子,执行 main 方法控制台输入模块表名回车自动生成对应项目目录中
    14. public class CodeGenerator {
    15. /**
    16. *

    17. * 读取控制台内容
    18. *

    19. */
    20. public static String scanner(String tip) {
    21. Scanner scanner = new Scanner(System.in);
    22. StringBuilder help = new StringBuilder();
    23. help.append("请输入" + tip + ":");
    24. System.out.println(help.toString());
    25. if (scanner.hasNext()) {
    26. String ipt = scanner.next();
    27. if (StringUtils.isNotBlank(ipt)) {
    28. return ipt;
    29. }
    30. }
    31. throw new MybatisPlusException("请输入正确的" + tip + "!");
    32. }
    33. public static void main(String[] args) {
    34. // 代码生成器
    35. AutoGenerator mpg = new AutoGenerator();
    36. // 全局配置
    37. GlobalConfig gc = new GlobalConfig();
    38. final String projectPath = System.getProperty("user.dir");
    39. gc.setOutputDir(projectPath + "/src/main/java");
    40. gc.setAuthor("nzc");
    41. gc.setOpen(false);
    42. // gc.setSwagger2(true); 实体属性 Swagger2 注解
    43. mpg.setGlobalConfig(gc);
    44. // 数据源配置
    45. DataSourceConfig dsc = new DataSourceConfig();
    46. dsc.setUrl("jdbc:mysql://localhost:3306/task1?useUnicode=true&useSSL=false&characterEncoding=utf8");
    47. // dsc.setSchemaName("public");
    48. dsc.setDriverName("com.mysql.cj.jdbc.Driver");
    49. dsc.setUsername("root");
    50. dsc.setPassword("123456");
    51. mpg.setDataSource(dsc);
    52. // 包配置
    53. final PackageConfig pc = new PackageConfig();
    54. pc.setModuleName(scanner("模块名"));
    55. pc.setParent("com.nzc");
    56. mpg.setPackageInfo(pc);
    57. // 自定义配置
    58. InjectionConfig cfg = new InjectionConfig() {
    59. @Override
    60. public void initMap() {
    61. // to do nothing
    62. }
    63. };
    64. // 如果模板引擎是 freemarker
    65. //String templatePath = "/templates/mapper.xml.ftl";
    66. // 如果模板引擎是 velocity
    67. String templatePath = "/templates/mapper.xml.vm";
    68. // 自定义输出配置
    69. List focList = new ArrayList<>();
    70. // 自定义配置会被优先输出
    71. focList.add(new FileOutConfig(templatePath) {
    72. @Override
    73. public String outputFile(TableInfo tableInfo) {
    74. // 自定义输出文件名 , 如果你 Entity 设置了前后缀、此处注意 xml 的名称会跟着发生变化!!
    75. return projectPath + "/src/main/resources/mapper/" + pc.getModuleName()
    76. + "/" + tableInfo.getEntityName() + "Mapper" + StringPool.DOT_XML;
    77. }
    78. });
    79. cfg.setFileOutConfigList(focList);
    80. mpg.setCfg(cfg);
    81. // 配置模板
    82. TemplateConfig templateConfig = new TemplateConfig();
    83. templateConfig.setXml(null);
    84. mpg.setTemplate(templateConfig);
    85. // 策略配置
    86. StrategyConfig strategy = new StrategyConfig();
    87. strategy.setNaming(NamingStrategy.underline_to_camel);
    88. strategy.setColumnNaming(NamingStrategy.underline_to_camel);
    89. strategy.setEntityLombokModel(true);
    90. strategy.setRestControllerStyle(true);
    91. // 公共父类
    92. // 写于父类中的公共字段
    93. strategy.setSuperEntityColumns("id");
    94. strategy.setInclude(scanner("表名,多个英文逗号分割").split(","));
    95. strategy.setControllerMappingHyphenStyle(true);
    96. strategy.setTablePrefix("pms_");
    97. mpg.setStrategy(strategy);
    98. mpg.setTemplateEngine(new VelocityTemplateEngine());
    99. mpg.execute();
    100. }
    101. }

    运行代码:

    执行结果

    默认里面是没有方法的话,不过用来写个基本的小demo是非常方便的。

    3.3、业务代码

    controller、mapper、entity都不是重点,一笔带过了哈。

    1. @RestController
    2. @RequestMapping("/category")
    3. public class CategoryController {
    4. @Autowired
    5. ICategoryService categoryService;
    6. /**
    7. * 查询出所有的数据,以树形结构组装起来
    8. * @return
    9. */
    10. @GetMapping("/tree")
    11. public Result listTree() {
    12. List entities= categoryService.listWithTree();
    13. return ResultUtil.success(entities);
    14. }
    15. }
    1. public interface CategoryMapper extends BaseMapper<Category> {
    2. }
    1. @Data
    2. @EqualsAndHashCode(callSuper = false)
    3. @TableName("pms_category")
    4. public class Category implements Serializable {
    5. private static final long serialVersionUID = 1L;
    6. @TableId(value = "cat_id", type = IdType.AUTO)
    7. private Long catId;
    8. private String name;
    9. private Long parentCid;
    10. private Integer catLevel;
    11. private Integer showStatus;
    12. private Integer sort;
    13. private String icon;
    14. private String productUnit;
    15. private Integer productCount;
    16. @TableField(exist = false)
    17. private List<Category> categoryChild;
    18. }
    1. public interface ICategoryService extends IService<Category> {
    2. /**
    3. * 封装成三级菜单形式
    4. * @return
    5. */
    6. List listWithTree();
    7. }

    真正的业务是在 serviceImpl 中

    1. @Service
    2. public class CategoryServiceImpl extends ServiceImpl, Category> implements ICategoryService {
    3. // @Autowired
    4. // CategoryMapper categoryMapper;
    5. @Override
    6. public List<Category> listWithTree() {
    7. //1、查询出所有的分类
    8. List<Category> allCategory = baseMapper.selectList(null);
    9. //2、组装成父子的树形结构
    10. //2.1、找到所有的一级分类
    11. List<Category> categoriesLevel1 = allCategory.stream()
    12. // filter 方法就如名称一样,就是用来过滤的
    13. // 满足条件的会留下,到最后会返回一个流式对象
    14. .filter(category ->
    15. category.getParentCid() == 0
    16. )
    17. // map 的理解百话点说就是可以对传入的对象做出修改
    18. // 这里的意思就是找出当前一级分类中的二级分类,然后再set进去
    19. // 最后再返回一个流式对象
    20. .map((menu) -> {
    21. menu.setCategoryChild(getChildrens(menu, allCategory));
    22. return menu;
    23. })
    24. // 见名知意,这就是排序 ,其实也算是重定义Java比较器
    25. .sorted((menu1, menu2) -> {
    26. return (menu1.getSort() == null ? 0 : menu1.getSort()) - (menu2.getSort() == null ? 0 : menu2.getSort());
    27. })
    28. // 就是将流转换为带泛型的List对象
    29. .collect(Collectors.toList());
    30. return categoriesLevel1;
    31. }
    32. private List<Category> getChildrens(Category root, List<Category> all) {
    33. // 这里的逻辑和上面一段是一样的,这里是构造三级、四级等等多层也是一样构造的
    34. List<Category> collect = all.stream()
    35. .filter(category -> {
    36. return category.getParentCid().equals(root.getCatId());
    37. }).map((menu) -> {
    38. menu.setCategoryChild(getChildrens(menu, all));
    39. return menu;
    40. })
    41. .sorted((menu1, menu2) -> {
    42. return (menu1.getSort() == null ? 0 : menu1.getSort()) - (menu2.getSort() == null ? 0 : menu2.getSort());
    43. })
    44. .collect(Collectors.toList());
    45. return collect;
    46. }
    47. }

    还有一些公共代码和配置文件什么的,在仓库中都是有的,这里不再重复贴出来啦。

    3.4、测试结果

    我就是使用 Postman 测试的,没有专门画前端,前端太弱鸡啦,求大佬们放过...

    源码:github,gitee

    三、总结

    真正到了工作的时候,你会发现你写代码和平常还是会有很大差别的,最主要就是体现在业务场景这个方面。

    写 Demo 的时候,主要就想着能写出个 Demo 就不错了,但实际上真到了该上业务的时候,综合起来还是有很多值得思考的问题。

    在学习的时候针对一些必要的技术还是可以往深里挖掘挖掘的,并且自己也要学会根据现学的东西想到应用场景,思考如何整合项目,那些地方还有不足的,项目的扩展性如何等等,这些可以不用实现,但是在学习的时候能够多一些思考,这对于学习一个新技术是很有必要的

  • 相关阅读:
    系列七、栈 & 堆
    接口测试vs功能测试
    文字悬停效果
    4.力扣c++刷题-->删除有序数组中的重复项 II
    GEE——利用map函数获取指定时间范围内年份月份日期内的所有影像并求降水平均值
    曼孚科技入选IDC中国数据智能市场代表厂商
    DS线性表—多项式相加 C++
    SpringCloud 客户端负载均衡:Ribbon
    快速入门C++正则表达式
    RocketMQ生产者原理及最佳实践
  • 原文地址:https://blog.csdn.net/weixin_62421895/article/details/126118026