1.情景展示

  在JavaWeb开发过程中,我们通常会基于springmvc分层思想对整个项目进行分层开发:

  常见的就是分为model(域模型层)、dao(数据库访问层)、service(业务逻辑层)、controller(控制器层)、web(表现层),这样分层之后,各个层之间的职责会比较明确,后期维护起来也相对比较容易。

  通常我们会将其维护到一个模块当中,也就是一个项目就是一个模块,把不同的层用包进行区分,如下图左侧部分。

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发

 

  但是,随着项目越来越复杂,每个层的规模也逐渐增大,在一个模块中进行开发,会给测试和维护带来不便,对于大型项目来说,一般会将每个层放到自己的模块中,然后每个层建立联系,单独维护。 

  对于后期开发维护人员来说,右面这种分模块的方式更直观,看着比单模块舒服很多。

  好处不多说,用了才知道。 

2.分层梳理

  下面两张图片方便大家对于即将分开的模块之间的关系有着重要的指导意义。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_02

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_03

  按照上面的思路,确定好哪些包需要移到哪些模块,下面就着手迁移啦。

3.分模块搭建/改造

  新建模块

  选中项目,右键--》New--》Module

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_04

  下一步 

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_05

  给模块起个名字

  关于名字,如果项目名称不长的话,建议使用“项目名称-模块名称”的方式,这样更加直观;

  当然了,这是建议,直接取模块名称也是完全没有问题的。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_06

  我们先来了解一下,新建模块的结构:

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_07

  说白了,这就是maven的标准目录(因为还没有Java文件,所以还没生成target目录,idea会帮我们完成的)。 

第一步:迁移公共类(如果没有就忽略);

  base目录,我放的是基本工具类等公共需要用到的东西

  将这个base目录直接拖拽到bill-base模块java目录下

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_08

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_09

  在To package里输入包名路径,路径构成一般为:

  com.公司名称.项目名称.原来的包名,这有这样,才能保证不同模块的路径前缀是一致的。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_10

  不存在的包名会被创建

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_11

  创建成功后,class连同包名都会完成迁移

 

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_12

  如果出现上面这种原来的被没有删除的情况,手动删除即可。

  我们知道,基本工具类一般情况下,是需要jar包依赖的,迁移成功后如果报错的话:

  鼠标悬浮上去,在idea中,点击添加jar包,会自动将jar包依赖增加到对应的pom文件中。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_13

  注意:

  由于是基础工具类,其它模块不一定会用到这个模块,即使需要用到,也不一定就需要用这里面的jar包,所以,我们可以把bill-base里的依赖设置成可选依赖,这样,其它模块需要对应jar包时,需要重新引入依赖;也不宜引起jar包冲突。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_14

第二步:迁移实体类;

  新建model模块

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_15

  同样,关键在于迁移的包名(要把po这个包放到哪里) 

  第二关键点:下图提示,是在告诉我们,哪里用到了po包里的Java类

  不要理会,强制迁移。 

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_16

  迁移成功

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_17

  在这里,po层就显得有些多余了,重命名,将其删除即可。 

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_18

第三步:迁移dao层;

  新建dao模块;

  迁移dao层;

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_19

  强制迁移;

  如果出现多余的包名,通过重命名将其删除。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_20

  我们知道,dao层又叫持久层,SQL映射也在这层完成,所以,我们只迁移dao层是远远不够的,还可以将对应的SQL.xml迁移过来。

迁移SQLMap.xml文件;

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_21

  拖拽迁移,并新建一个mapper文件夹

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_22

  容易出错的地方:

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_23

  错误目录长这个样子,最终导致在启动项目后,对应SQL无法执行的问题。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_24

  如果出现删除原来目录导致新生成的目录也随之消失的情况,只能还原,只拖动xml完成迁移了。 

  更改sqlMapper.xml的命名空间(与dao层相对照)

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_25

  改对了,下面就不会再报红了。 

还没结束,还需要完成jar包依赖的迁移;

  首先是模块依赖;

  按照依赖对照关系,dao肯定需要依赖model。所以,在bill-dao的pom.xml中添加对model的依赖。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_26

  补充:在子模块中,我们可以指定父pom文件访问路径,这样方便我们跳转及识别。

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_27

  其次才是jar包依赖。 

  将父pom.xml中的关于mybatis,数据库连接池,mybatis-plus等相关jar包依赖 

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_28

  重新导包

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_29

最后一步:重新编译项目。

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_30

  注意:这里,只重新编译bill-dao模块,因为还没有迁移完毕,重新构建整个项目的话,报错信息会出现一大坨,但我们根本无法修改。

  说明:要想只编译bill-dao模块,需要鼠标通过点击选中这个模块,再进行Build选项时才会出现编译这个模块的选项。

  你会发现一堆bill-dao模块的报错信息,不要怕。

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_31

  双击错误信息,会直接跳转到报错的java类。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_32

  改成正确引用路径即可。

  挨个把所有bill-dao模块的报错信息改完就行了。

第四步:迁移service层;

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_33

  如果我们像往常一样进行迁移

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_34

  这里会出现一个问题:

  如果不是新建的目录(一个不存在的目录)的话,将会迁移失败。

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_35

  必须这样做,才能迁移成功:

  选中java目录,右键新建package

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_36

  选中service下的所有子包。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_37

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_38

  强制迁移。 

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_39

  迁移成功,但vo包没有迁移过去。

  引入模块依赖,导包;

  构建bill-service模块

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_40

  这里分两种情况:

  一种是向上依赖,像原来的dao层和po层已经被我们迁移了出来,改下路径就行了;

  另一种是向下依赖,service层依赖web层,这种向下串层的行为在模块化后是不被允许的,

  像:模块A依赖模块B,模块B又依赖于模块A,这种互相依赖是不被允许的,强制引入,在最终打包的时候会导致打包失败;只能进行单向或单方依赖。

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_41

  所以,解决办法就是把service层需要的web层的相关包全部迁移到service模块。 

  迁移完成

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_42

  这里需要注意的是:

  按理说,dto是需要放在web模块的,但是,由于service模块需要用到,所以,我只将dto里部分文件进行了迁移(最好不要这样搞)。

  迁移jar包依赖;

  大部分的jar包依赖都集中在service模块。

  这里需要提醒的是:

  虽然校验和knife4j严格意义上来说,是属于web模块的东西,但是,web和service两个模块其实是不能完全撇清的,剪不断理还乱。

  既然service层要用,索性干脆把这类jar包全部从web层搬到service层,这样,web模块在引入service模块后,也就引入了这些jar包依赖。

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_43

  对于程序的运行,不受丝毫影响。 

  完成jar包迁移后,重新导包,重新编译bill-service模块,也许就会报错了。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_44

  如果项目中使用了Lombok插件,就会报错。

  原因是:我们刚才将Lombokjar包依赖从父pom文件中迁移到了bill-service子pom文件,身为同级的bill-model自然就找不到啦。

  所以,我们需要将lombok依赖从bill-service,迁移到bill-model。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_45

  前面也说了,只要其它同级模块引入这个模块,其它模块就也能直接使用Lombok注解,无需重新引入jar依赖。 

第五步:迁移web层。

  鉴于上面相同目录导致迁移失败的问题,我们还是手动创建包名好了。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_46

  总之,在idea中进行迁移会引发各种各样的路径问题,真是能恶心死人,只能自己逐个排查或这等报错再改啦。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_47

  迁移完毕

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_48

  bill项目下的src变得空空如也,直接将这个src删除即可。 

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_49

  也许,我们的项目所有模块都会变成这个吊样,真是想骂街:

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_50

  都被强制加一个web包。 

  只能通过重命名后再将web目录删除。

  需要提醒一点:

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_51

  启动类是放在项目根目录下的,即:和web同级,别像我一样,改着改着改懵了,把它们也放到了web目录下,

  导致的结果就是:

  启动项目后,一直报错:找不到注入的service模块类,A component required a bean of type... 

  添加模块依赖;

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_52

  说明:

  第一,引入的模块,必须指定版本号,不能省略<version>标签;

  第二,版本号的值可以跟随项目通过,通过${project.version} 取值。

  迁移jar包;

  把父pom文件中所剩下的jar包依赖全部迁移到bill-web的pom文件中(父pom.xml中最好一个jar包依赖都不要留)。 

  bill-web的pom文件,除了模块依赖,通常情况下只有单元测试和springboot相关依赖。

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_53

  重新编译bill-web目录,解决报错路径问题。

修改启动类注解扫描包;

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_54

  dao路径改成正确的

修改日志配置文件关于dao层的引用

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_55

修改拦截器关于控制层路径引用

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_56

迁移项目构建工具;

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_57

将父pom.xml打包方式设为pom;

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_58

  项目公共属性配置,也在父pom文件中进行,子pom文件通过${}引入即可。 

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_59

  如果需要多环境支持,在父pom中添加profile即可。 

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_60

  父pom文件,还可以配置maven中央仓库

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_61

  模块出错

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_62

  如果创建出来的模块或者在迁移过程中导致java或resources目录发生了变化:变成了普通目录,就需要将该模块删除,重新创建。 

  删除模块

  第一步:remove module

  选中要删除的模块,右键,选择“Remove Module”。

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_63

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_64

  从项目移除成功后,文件夹将会变色;

  右键--》删除即可。

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_65

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_66

  但是,如果再次重建这个模块时(模块名称一致),出现模块已存在的情况时,可以这样做:

  关闭项目,然后从磁盘中删掉项目,从svn或git上重新将整个项目下载下来再重新新建模块(重头再来)。

4.关于View层能否从Web层剥离出来的探究

  剥离步骤:

  第一步:创建view模块,并将web/main目录下的webapp目录迁移到view/main下

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_67

  第二步:web模块引入view模块依赖 

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_68

此时,要跳转的页面路径已经找不到了。

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_69

  第四步:重新导包并启动项目;

  项目启动正常

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_70

  浏览器访问跳转到jsp的请求

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_71

springboot 多模块分层开发、改造(搭建、运行、打包)_WEB开发_72

 

  经过以上测试发现:无法将前端界面从web模块进行剥离。

也就是说:要想由控制层Controller来决定最终跳转的页面的话,页面和Controller需要在同一个模块里。

5.运行

  改造完毕后,还是像不分模块之前那样,直接用main方法启动类启动项目即可。

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_73

  根本不需要所谓的在pom.xml中指定唯一的main方法入口。

  可能会出现这种情况:

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_74

  解决办法:

  点击这个启动类,选择“Edit Configurations”

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_75

 

  打开配置页面

  在这里,我们会看到底部错误信息提示:在bill模块中找不到这个启动类。

  实际上,bill是一个项目而不是模块,我们只需要将模块指定成该启动类所在模块即可。

  展开“Environment”界面,在“Use classpath of module”中进行模块切换即可。

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_76

  如果能成功启动还能看到原来的执行效果,那么项目多模块的重构就完成了。

(通过idea,重构的多模块项目,还有一个需要修改的地方,见第七部分)

6.打包

  要想打包,需要确定三个问题:

第一,打成什么样的包(war还是jar)?

第二,哪个模块作为主模块(不是父模块),之所以这样叫,是因为,其它同级或子级模块都将会被它包含在该模块中。

  对应前后端交互的项目,我们通常会将web模块作为主模块进行打包。

  在web模块对应的pom.xml中指定packaging,如果设置该标签,与普通项目一样,将默认被打成jar包。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_77

  添加构建工具;

  指定构建工具,并引入springboot自带的maven插件。

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_78

  注意:其它模块也是允许有自己的构建工具。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_79

  虽然,service模块也指定了maven编译插件,但是,web还是主模块,并没有受到影响。

  这就让我很疑惑了,idea是怎么选取web作为主模块的?

  为了测试这个问题,我将dao里也加上了springbootmaven插件

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_80

  打包失败:dao模块没有程序入口

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_81

  通过以上实验,大致了解到: 

  当pom.xml引入springbootmaven插件后,在打包时,该插件会在该模块中寻找启动类。将程序入口作为主模块也就成了顺理成章之事了。

  所以,一般情况下,引入springbootmaven插件的模块将会被作为主模块。

第三,选择哪个模块进行打包?

  通过上面得知:

  既然,web会被识别为主模块,那我们直接对它进行打包不就行啦?

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_82

  执行package命令,结果却打脸:

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_83

  web模块所依赖的这些同级模块,竟被从maven仓库中下载它们所对应的jar包,结果可想而知。

  所以,不能用主模块完成整个项目的打包。 

  在idea中,通过一个带(root)的maven模块完成项目打包。

  也只有通过它才能完成整个项目的打包,下面的子模块只能完成对自己模块的打包。

  通过root会分别对每个模块进行打包,这是与没有分模块开发的项目,在打包时最大的区别。

  打开右侧的maven视图

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_84

  打jar包

  在web模块对应的pom中指定成jar包。

  clean,package 

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_85

  日志输出如下: 

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_86

  jar包展示

springboot 多模块分层开发、改造(搭建、运行、打包)_SpringBoot_87

  打war包

  方式同上

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_88

  war包展示

springboot 多模块分层开发、改造(搭建、运行、打包)_jar包_89

  我们可以看到,其它模块都被打成了jar包,因为我们没有在对应的模块的pom文件中指定打包形式。

  我觉得这样挺好的,提高了代码的安全性,让别人不能通过项目一下就全揽了所有代码。

  如果你还配置了多环境,勾选上指定环境再打包就可以了。

springboot 多模块分层开发、改造(搭建、运行、打包)_maven_90

7.多模块引发的404问题

2022年2月24日17:32:15

使用idea将项目分离成多模块项目后,当我们去访问webapp目录下所有的文件时,都将得到一个结果:404。

springboot 多模块分层开发、改造(搭建、运行、打包)_xml_91

解决办法,见文末推荐。

当然,如果你不涉及需要访问网页的话,可以不用管这个问题。

写在最后

  哪位大佬如若发现文章存在纰漏之处或需要补充更多内容,欢迎留言!!!


作者:Marydon