包引用与依赖管理
1.17新特性:module依赖图修剪与延迟module加载
在Go 1.17及之后版本,在执行命令go mod tidy时,Go module 会自动整理go.mod 文件,如果有必要会在部分依赖包的后面增加// indirect注释。一般而言,被添加注释的包肯定是间接依赖的包,而没有添加// indirect注释的包则是直接依赖的包(所谓直接依赖的包,即明确的出现在某个import语句中,直接在项目代码里搜能搜到的)
1.17后一般go.mod至少会有两个require,第一个是直接依赖,另一个是间接依赖
如何查找间接依赖来源:
Go module提供了go mod why命令来解释为什么会依赖某个软件包,若要查看go.mod中某个间接依赖是被哪个依赖引入的,可以使用命令go mod why -m <pkg>来查看
go mod why -m all则可以分析所有依赖的依赖链
Tony Bai–Go 1.17新特性详解:module依赖图修剪与延迟module加载
go.mod中// indirect有什么作用?
go.mod文件中的// indirect注释用于说明某些模块依赖是间接的。
具体来说:
当一个模块直接require其他模块时,这个依赖就是直接依赖项,不需要标注
但是如果一个模块A依赖模块B,模块B又依赖模块C,那么对模块A来说,模块C就是间接依赖项
在go.mod中,我们通常只关心直接依赖,间接依赖不需要显式声明
但是间接依赖也会计算到依赖图和模块版本匹配中
为了明确表示这是间接依赖关系,go.mod支持在依赖行后添加// indirect注释
这样其他开发者一目了然知道这是通过其他模块传递而来的间接依赖
所以// indirect注释的作用是:
明确标注某个依赖是通过其他模块传递而来的间接依赖
区分直接声明的依赖和间接引入的依赖
让依赖关系更清晰明了
它并不影响模块的编译或版本对齐,只是一种注释说明依赖来源的约定。
间接依赖项是否会被计算在模块的依赖图中?
是的,go.mod中的间接依赖也会被计算在模块的依赖图中。
go.mod会递归地分析依赖关系,不仅直接引用的模块会被记录,还包括这些模块引用的次级依赖,以此类推。
举个例子:
1 | module A |
1 | module B |
这样,虽然模块A只直接依赖B,但在解析依赖关系时,C也会被包含进来,成为A的间接依赖。
go toolchain在操作依赖时(比如go get更新依赖),会自动解析并处理整个依赖图,包括间接依赖。
间接依赖也会影响模块版本的选择。例如在模块B和C都升级到v2.0.0后,模块A还停留在v1.0.0时,可能会导致版本冲突。
所以正确管理间接依赖也是模块版本管理的一个重要方面。
总之,go.mod的依赖关系会递归地计算出完整的依赖图,这对依赖解析和版本管理都很重要。
go.mod中的间接依赖会如何影响模块的构建和编译过程?
go.mod中的间接依赖会对模块的构建和编译过程产生以下影响:
- 构建路径:
在编译一个模块时,go toolchain会根据go.mod计算出完整的依赖关系图,这决定了包的导入路径。
所以间接依赖也会影响包的导入路径,必须将间接依赖的包导入路径也包含进去,否则编译会失败。
- 依赖版本:
间接依赖也必须使用兼容的版本。如果间接依赖的版本和直接依赖声明的版本不一致,会导致编译错误。
- 替代方案:
如果一个间接依赖有多个可替代的模块(例如同一个仓库的fork),编译可能会根据环境不同采用不同的替代模块。
- 编译优化:
go toolchain在编译时会重用不变的包,间接依赖的包也会加入到这个重用逻辑中去。
- vendoring:
如果启用vendoring模式,所有的依赖包括间接依赖都会被写入vendor目录中。
总之,间接依赖会递归地影响依赖的解析和编译过程。正确处理间接依赖是组件化和依赖管理的重要一环。
所以go.mod必须描述完整的依赖关系图,才能保证模块在不同环境下的编译一致性。
如何查看一个模块的完整依赖图?
可以使用 go mod graph 命令来查看 Go 模块的完整依赖图。
基本语法是:
1 | go mod graph |
这会打印出当前模块递归的整个依赖关系图。
例如:
1 | go mod graph |
可以看到直接依赖和间接依赖的关系。
也可以用 DOT 语言输出,然后将其可视化:
1 | go mod graph | dot -Tpng -o graph.png |
此外,go mod graph 命令还有一些额外的选项:
- -json 输出 JSON 格式的依赖图
- -dot 输出 DOT 语言描述的依赖图
- -test 是否包括测试依赖
- -min minimal version selection (最小版本选择)
以上命令可以帮助我们分析复杂项目的依赖关系,找到依赖冲突的原因,或者查看不同版本依赖差异。
总之,go mod graph 是查看模块依赖图非常有用的工具。
go.mod中的// indirect注释是否会影响模块的依赖解析和版本管理?
go.mod文件中的// indirect注释不会影响模块的依赖解析和版本管理。它仅仅是对依赖关系的一个注释或说明,没有实际作用。
具体来说:
模块的依赖解析和版本匹配是根据所有直接和间接依赖进行的,不分 direct 或 indirect
模块中的// indirect注释纯粹表明依赖来源,不影响具体依赖关系图的构建
即使去掉或重新增加一个// indirect注释,依赖解析和版本匹配不会有任何改变
go命令执行升级、下载、构建等操作时也不会特殊处理带有// indirect注释的依赖
它只是对开发者提供一个依赖来源的友好提示,并不属于依赖声明的实质部分
所以总结来说:
// indirect注释纯属说明性,不可改变模块真正的依赖关系
依赖解析和版本匹配完全依照所有直接和间接依赖来完成
go命令不会根据// indirect来区分对待不同类型的依赖
它不具备影响依赖管理的任何实质作用,只是一种注释标识依赖来源关系。
go命令如何处理直接和间接依赖之间的冲突?
go命令在处理直接和间接依赖之间可能出现的冲突时,有如下规则:
直接依赖优先级高于间接依赖。也就是说,如果直接依赖和间接依赖之间版本不匹配,将选择直接依赖版本。
如果一个模块直接和间接声明了同一个依赖但版本不同,这是冲突的,go命令会报错。
如果直接依赖声明了一个版本,间接通过其他模块引入了不同版本,go命令不会报错,但是会警告版本不匹配。
如果移除造成不匹配的直接依赖声明,则间接依赖的版本将被采用。
如果移除直接依赖后,间接依赖路径上也存在版本不匹配,则go命令同样会报错。
处理流程如下:
直接依赖优先,检查直接依赖版本是否匹配
如果不匹配,尝试使用间接依赖版本
检查间接依赖版本是否也存在不匹配
如果存在不匹配无法选择唯一版本,报错
否则使用匹配版本并打印警告
所以总体规则是:直接依赖优先,出现冲突直接报错,可以通过修改直接依赖消除冲突。
原文链接: https://dashen.tech/2022/06/21/Go-Modules相关参考/
版权声明: 转载请注明出处.