Maven,学习框架之前我们都会接触到的一个工具,感觉他的定位,似乎就跟git一样,只是方便我们开发?于是自然而然的,很多小猿对于Maven都只是停留在会用的阶段,利用他来构建,打包,引入jar包。
而实际上呢,Maven还有很多强大的地方,多模块开发,私服等等也是需要我们掌握的,在实际的开发中,往往是多模块共同开发,特别是我们之前提及的Dubbo分布式开发中,多模块开发是重中之重!
传统项目开发 导jar包- 既然是导jar包,我们是在本地windows开发,而实际项目是要放到linux服务器上,如果我们 单纯 把项目打成jar包丢上去,依赖文件也是jar包,但两个系统因为环境不一样,直接丢jar包(他不会重新编译运行的),得出来的结果可能就不一样
比如String的getByte方法,在两个系统上得到的结果是不一样的,如果 没有重新编译运行 ,那最终也会有偏差
Maven概念 POMProject Object Model
- 把每个项目看成一个对象来进行管理
提供一套自动化构建项目的方法.并且通用,兼容性好,跨平台
- 包括打包,编译,测试,运行等一套操作下来,让你在开发环境也可以方便进行测试等功能
构建是面向过程的,就是一些步骤,完成项目代码的编译,测试,运行,打包,部署等等。
maven支持的构建包括有:
1. 清理 , 把之前项目编译的东西删除掉,我新的编译代码做准备。
2. 编译 , 把程序源代码编译为执行代码, java-class文件
- 批量的,maven可以同时把成千上百的文件编译为class。
- javac 不一样,javac一次编译一个文件。
3. 测试 , maven可以执行测试程序代码,验证你的功能是否正确。
4. 报告 , 生成测试结果的文件, 测试通过没有。
5. 打包 , 把你的项目中所有的class文件,配置文件等所有资源放到一个压缩文件中。
- 这个压缩文件就是项目的结果文件, 通常java程序,压缩文件是jar扩展名的。
- 对于web应用,压缩文件扩展名是.war
6. 安装 , 把5中生成的文件jar,war安装到本地仓库供别的文件使用
7. 部署 , 把程序安装好可以执行。
这些在下边的生命周期其实刚刚好体现出来
依赖管理- 处理jar包冲突问题
即能进行配置的不要去编码指定,能事先约定规则的不要去进行配置。这样既减轻了劳动力,也能防止出错。
- 实际上maven并没有强行要求约束我们项目的文件结构,而是因为他做得好,自然而然就变成一种通用的结构了
先在本地仓库找,找不到就去私服找,如果私服没有,则会到中央仓库下载 到私服 并且发送到本地仓库
无论如何都还是最终会下到本地,区别只是从哪发送到本地而已?
- 此处有点疑问,私服没有会去中央仓库下到私服吗?我下边配置了阿里云和私服,似乎私服没有的话是直接从阿里云 直达下载 到本地仓库,并不会通过私服?希望大佬可以指正一下!
图源 Maven访问仓库顺序是什么 - 开发技术 - 亿速云
镜像仓库配置 mirrorOf对哪种仓库进行镜像
手工mavenMaven工程目录结构
构建命令Maven构建命令使用mvn开头,后面加功能参数,可以一次执行多个命令,使用空格分隔
clean把编译的去掉,target文件目录删除掉
install把你的当前项目安装到本地仓库,成为一个jar包供别的项目引用 里边的类 等
package会先compile,然后再clean
依赖管理 依赖传递依赖具有传递性,包括直接传递和间接传递。
直接传递:在当前项目中通过依赖配置建立的依赖关系(A使用B,A和B就是直接传递)
间接传递:如果A依赖B,而B依赖C,那么A和C之间就是间接传递
冲突原则 路径优先在越浅层的优先级越高
声明优先在同一层,先声明的比较高
在同一个pom中配置了相同资源的不同版本后配置的覆盖前边的
可选依赖 排除依赖 ①手动排除依赖,,且2可以直接把3整个依赖进来,再排除掉特定的 ②借助idea插件排除依赖 -- Maven Helper打开pom文件,下方就可以选择切换查看方式
右键选中你需要排除的 Exclude即可
③版本锁定-- (待补充实例,拿Dubbo的吧)Dubbo的GitHub文档中应用到了这点
2.3.0.RELEASE 2.7.8 org.springframework.boot spring-boot-dependencies${spring-boot.version} pom import org.apache.dubbo dubbo-dependencies-bom${dubbo.version} pom import org.apache.dubbo dubbo-spring-boot-starter2.7.8
dependencyManagement
- 通过它元素来管理jar包的版本,让子项目中直接引用依赖而不用列出版本号。Maven会 沿着父子层次向上走 ,直到找到一个拥有 元素的项目,然后它就会使用在这个dependencyManagement元素中指定的版本号。
管理通用性
- 我们不需要给每个子项目都去声明版本了,只需要在 最顶层父类 中统一进行管理。
可扩展性
- 如果某个子项目需要另外一个版本号时,只需要在dependencies中声明一个版本号即可。子类就会使用子类声明的版本号,不继承于父类版本号。
通过继承可以实现在子工程中沿用父工程中的配置(与Java类似)
制作方式 在子工程中声明其父工程坐标与对应的位置
父工程pom文件地址
如SpringBoot中
org.springframework.boot spring-boot-starter-parent2.5.6
在父工程中定义依赖管理可以管理版本,解决子工程中依赖冲突问题(上文提及到的版本锁定)
继承依赖使用:在子工程中定义依赖关系,无需声明依赖版本,版本参照父工程中依赖的版本
聚合- 我们实际开发中,很多时候是多模块开发,会把业务单独抽离出来成为一个模块,比如 Dao 模块, Service 模块
- 那这个时候比如说 Dao 模块发生了变化,其他所有依赖 Dao 模块的模块怎么办呢,都得去更新吗?他自己知道吗?还是得我们自己去重新install重新依赖,有没有办法让他们 同时更新 而不是单独更新呢?--这就要用到我们的聚合
聚合module 效果 私服仓库分类pom
- 宿主仓库hosted:指的是我们公司或团队内部的包,并且 hosted 类型的仓库会分为 releases 和 snapshots 两个,前者是正式版,后者一般是开发测试版(快照版);
保存那些无法从中央仓库直接获取的资源,比如说我们自己研发的一些项目,或者一些第三方非开源项目比如oracle(要注意版权问题)
- 代理仓库proxy:用来代理中央仓库,例如我们依赖的包在本地仓库没有,就会到私服获取,私服没有的话,会到中央仓库先把包下载到私服,然后再下载到本地仓库;
- 仓库组group:把多个仓库组合起来,然后我们项目中只需要配置上这个类型的仓库地址,就可以把它里面组合的几个仓库都关联上.
有时候我们是多台机器开发,而我们要去引用别人机器上的模块jar包时,这个时候对方单纯install也只是发布到对方电脑上的本地仓库而已,我们还是访问不到.这个时候就需要一个中间商私服,来存放一些需要 共享的资源
nexus下载安装- 首先要下载 nexus 搭载我们的私服
- 下载完成后,修改etc中的启动端口(默认是8081)
- 进入bin目录
- vim nexus
- 进入后直接 /run_as_user 查找到run_as_root 再按 i 进入编辑模式
把这一项改为false , 才可以 用root用户开启nexus
- 修改完成后 按ESC退出编辑模式进入命令模式 输入 :wq 保存退出即可
- 最后 ./nexus start 启动
- 访问你设置的端口号
账号默认是admin
初次登录会提示你去特定位置找密码
在该目录下寻找即可
之后可以修改默认密码
访问私服仓库(手动上传)--不推荐- 上方可以切换视图,左边是浏览仓库,右边的管理设置仓库,我们在管理这一层添加一个自己的仓库
需要配置两个地方
- 一个是本地仓库如何跟私服打交道 -- 对应默认的 总setting文件
- 一个是我们的工程要发布到私服具体哪个仓库 -- 对应 当前项目pom文件 里边的配置
上传jar认证配置上传jar包需要 认证 ,maven的认证是在.m2/settings.xml或自己的的maven里 settings.xml 中servers标签下配置的.
这里特别说明一下,servers配置的ID很关键,这个与你项目里面的 distrubtionManagement 下配置的仓库ID一致,否则会报审核未通过的错误。
nexus-Melo
admin
admin
Melo-Release
admin
admin
Melo-Snapshots
admin
admin
这里的id对应了setting.xml里配置的server.id信息, name 任取
Melo-Release
http://ip地址:端口/repository/Melo-Release/
Melo-Snapshots
http://ip地址:端口/repository/Melo-Snapshot/
上传效果
idea环境从私服获取(同时使用阿里云镜像+私服)
- 配置全局setting.xml,我们所有项目就都会去私服找了
建议使用自带的maven-public仓库组 , 然后把我们 自己创建的仓库 包含进去
注意不要把proxy类的仓库包含进我们的group!这样我们配置的阿里云镜像就失效了除非自己去手动改proxy代理的仓库地址,但个人感觉不太推荐,具体见后文"另外的实现方式"
配置server--访问服务器的权限包含拉取和上传的两部分配置,两部分都需要权限
nexus-Melo
你的私服账号
你的私服密码
Melo-Release
你的私服账号
你的私服密码
Melo-Snapshots
你的私服账号
你的私服密码
配置mirror
alimaven
aliyun maven
http://maven.aliyun.com/nexus/content/groups/public/
central
nexus-Melo
nexus-Melo
!central
http://ip地址:端口/repository/maven-public/
扩展--mirrorOf
*表示所有 !表示非
,!repo1 : 在除了repo1的仓库中寻找
external:
: 刚好跟上边的 !internal.repo,* 相反
扩展--mirror**
- 默认情况下配置多个 mirror 的情况下,只有第一个生效。
- 无法连接的时候,才会去找后一个;而我们想要的效果是:当某个jar包在第一个mirror中不存在的时候,maven会去第二个mirror中查询下载,但是maven不会这样做!
正确的操作是在profiles节点下配置多个profile,而且配置之后 要激活!!!!
配置profile激活profile!!aliyun aliyun http://maven.aliyun.com/nexus/content/groups/public/ true true always nexus-pr nexus-Melo nexus-Melo http://ip地址:端口/repository/maven-public/ default true true
aliyun
nexus-pr
jdk-1.8
实现阿里云与私服共存的效果
那些中央仓库有的文件,就会 走阿里云镜像 去下载,如果是我们自定义的jar包(中央仓库没有的),就会 走我们的私服 去下载!
另外的实现方式- 其实我们也可以直接在私服创建一个proxy仓库,让他代理阿里云镜像就好了,但是这样其实也有个问题,就是proxy仓库本身是不允许我们自定义上传组件的,所以综合考虑的话还是上边自行配置的方法好一点
- maven这一块,具体的 聚合 , 继承 实例,等到真正运用上了分布式开发的时候,应该会有所渗透。而私服的运用,在多台机器上是十分重要的!



