Go modules基础精进,六大核心概念全解析(上)
点击一键订阅《云荐大咖》专栏,获取官方推荐精品内容,学技术不迷路!

Go 语言做开发时,路径是如何定义的?Go Mudules又为此带来了哪些改变?本文将会全面介绍Go modules六大核心概念,包括了设计理念与兼容性原则等,掌握这些技术点对于管理和维护Go 模块有重要价值。

上一篇文章中,笔者介绍了如何以经典的 hello world 为例创建一个 Go module 模块,需要说明的是一个模块中是可以包含多个包(package)的,它们是可以被一起发布、打包、版本化的。同时,Go Modules 也可以通过版本管理系统(github、gitlab)或者 goproxy 代理进行下载。在使用 Go Modules 之前,建议大家弄清楚息息相关的六大核心概念,以方便大家在后期的开发、使用过程中理解更加深入。
我们在使用 Go 语言做开发时经常会遇到像 “example.com/test” 或者 “example.com/test/pkg/log”这样的路径,这些路径到底是怎么定义的,两者中存在什么关系,在 Go Modules 中又扮演着怎样的角色呢?Go Modules 的引入对已有的包又引入了哪些新的概念,它们是如何协作的?对兼容性提出了哪些新的要求呢?让我们一起来看一下。
一:模块路径 (Module Path)
Go 使用 “module path” 来区分不同的 module 模块,它在 go.mod 文件中被定义,这个文件中还包含了这个模块编译所需的其他依赖。如果一个目录中包含了 go.mod 文件,那么这个目录就是这个 Go 模块的根目录了。
另外,还要介绍下包(package) 这个概念,它在 Go Modules 出现之前就已经存在了。Go 模块中的 “包 (package)”是处于同一目录中的一些源代码文件的集合,这些文件将被编译在一起。“包路径(package path)”是模块路径和子目录(模块根目录的相对路径)的组合。举个例子,在模块“golang.org/x/net”下的 html 目录中有个包,这个包的路径是 “golang.org/x/net/html”。
总结下来就是: 一个代码仓库可以包含多个 Go 模块,一个 Go 模块可以包含多个 Go 包。
模块路径是一个 Go 模块的规范名称,用于区分不通的模块。同时他还是该模块下 Go 包的路径前缀。理论上,模块路径应该至少包含两个关键信息:
模块的作用
哪里获取该模块
二:版本号与兼容性原则
版本号相当于是一个模块的只读快照,它可以是正式的发布版本,也可以是预发布版本。 每个版本都以字母 v 开头,后跟一个语义版本,例如 v1.0.0。
总而言之,语义版本由三个由点分隔的非负整数(主要版本、次要版本和补丁版本,从左到右)组成。 补丁版本后可以跟一个以连字符开头的可选预发布字符串。 预发布字符串或补丁版本后可以跟一个以加号开头的构建元数据字符串。 例如,v0.0.0、v1.12.134、v8.0.5-pre、v2.0.9+meta 等都是有效版本。
版本号中的信息代表了这个版本是否是一个稳定版,是否保持了与之前版本的兼容性。
- 当维护的模块发生了一些不兼容变更,比如修改了外部可调用的接口或者函数时,需要对主版本号进行递增,并且将次版本号和补丁版本号置为零。比如在模块中移除了一个包。
 - 在模块中添加一些新的函数或者接口,并没有影响模块的兼容性时,需要对次版本号进行递增,并且将补丁版本号置为零。
 - 当修复了一些 bug 或者进行了一些优化时,只需要对补丁版本号进行递增就可以了,因为这些变更不会对已经公开的接口进行变更。
 - 预发布后缀代表了这个版本号是一个预发布版本。预发布版本号的排序会在正式版本号的前面。举个例子,v1.2.3-pre 会排列在 v1.2.3 前面。
 - 元数据后缀会在版本比对中被忽略,版本控制中的代码库会忽略带有构建元数据的标签,但在 go.mod 文件中指定的版本中会保留构建元数据。如果一个模块还没有迁移到 Go Modules 并且主版本号是 2 或者更高,+incompatible 后缀会被添加到版本号上。
 
如果一个版本的主版本号是 0 或者它有一个预发布版本后缀,那么这个版本被认为是一个不稳定版本。通常,不稳定版本不受兼容性限制的,举个例子,v0.2.0 可能和 v0.1.0 是不兼容的,v1.5.0-beta 可能和 v1.5.0 也是不兼容的。
Go 可以通过 tags、分支、和 commit 哈希值来获取模块,即使这些命名没有遵循这些规则。在主模块中,go 命令会自动的将这些 revision 转化为符合标准的版本号,其被称为伪版本号(pseudo-version)。举个例子,当执行下面的命令时:
go get -d golang.org/x/net@daa7c041
Go 会讲指定的 hash daa7c041 转化为一个伪版本号 v0.0.0-20191109021931-daa7c04131f5。在主模块之外需要规范版本,如果 go.mod 文件中出现像 master 这样的非规范版本,go 命令会报错。
三:伪版本号
伪版本号是一种预发布版本号的格式,其中包含了指定的 commit hash 值。另外,对于没有打标签的代码库,也可以使用伪版本号来表明某个版本,它可以在正式发布某个版本之前方便的进行测试。举个例子,每个伪版本号都有三部分组成:
- 基本版本前缀(vX.0.0 或 vX.Y.Z-0),它要么源自修订版之前的语义版本标签,要么源自 vX.0.0(如果没有此类标签)。
 - 时间戳 (yyyymmddhhmmss),这是创建 commit 的 UTC 时间。 在 Git 中,这是 commit 提交时间。
 - commit 标识符 (abcdefabcdef),它是提交 commit 哈希的 12 个字符的前缀,或者在 Subversion 中,是一个用零填充的修订号。
 
在这三个部分之下,又分为以下多种情况
- 如果之前没有基版本,那么诸如 vX.0.0-yyyymmddhhmmss-abcdefabcdef 这样的伪版本号将被启用。主版本号 X 需要匹配模块的主版本号后缀。
 - 如果之前的基版本号是一个像 vX.Y.Z-pre 这样的预发布版本,那么 vX.Y.Z-pre.0.yyyymmddhhmmss-abcdefabcdef 将被采用。
 - 如果之前的基版本号是一个像 vX.Y.Z 这样的正式版本,那么 vX.Y.(Z+1)-0.yyyymmddhhmmss-abcdefabcdef 将被采用,举个例子,如果基版本号是 v1.2.3,伪版本号可能是 v1.2.4-0.20191109021931-daa7c04131f5。
 - 基于不同的基础版本号,多个伪版本号是有可能指向同一个 commit hash 的,在对一个低于已经存在的伪版本号打标签时,这种情况就会发生。
 
上面介绍的这种伪版本号携带了两个非常有用的信息:
1. 伪版本号会高于这些已经存在的基础版本号,但是会低于后面生成的其他伪版本号。
2. 有相同基础版本前缀的伪版本按时间顺序排序。
伪版本号不需要手动指定。很多 Go 命令可以接受一个 commit hash 或者分支名,然后自动将其转化为一个伪版本号(或者一个标签,如果存在的话)。例如:
go get -d example.com/mod@master
go list -m -json example.com/mod@abcd1234
在本篇中,我们介绍了模块路径、版本号与兼容性原则、伪版本号三大概念,而在下篇我们将会继续介绍Go Modules核心概念,包括主版本号后缀、解析包路径到模块路径的流程、go.mod 文件,敬请期待。另外,腾讯云 goproxy 企业版已经产品化,需要了解的同学可以点击这里。

李保坤往期精彩文章推荐:Go语言重新开始,Go Modules 的前世今生与基本使用

《云荐大咖》是腾讯云加社区精品内容专栏。云荐官特邀行业佼者,聚焦于前沿技术的落地及理论实践之上,持续为您解读云时代热点技术、探索行业发展新机。点击一键订阅,我们将为你定期推送精品内容。
Go modules基础精进,六大核心概念全解析(上)的更多相关文章
- 【Docker】容器、虚拟机与Docker概念全解析
		
导读 Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化.本文立足于新手,从容器和虚拟机两个大 ...
 - 从零打造在线网盘系统之Struts2框架核心功能全解析
		
欢迎浏览Java工程师SSH教程从零打造在线网盘系统系列教程,本系列教程将会使用SSH(Struts2+Spring+Hibernate)打造一个在线网盘系统,本系列教程是从零开始,所以会详细以及着重 ...
 - 阿里云Serverless应用引擎(SAE)3大核心优势全解析
		
软件发展到今,企业业务系统日趋复杂,开发一个业务系统需要掌握和关注的知识点越来越多.除实现业务逻辑本身,还需考虑很多非业务的基础技术系统:如分布式cache和队列.基础服务能力集成.容量规划.弹性伸缩 ...
 - Oracle静态监听与动态监听概念全解析
		
基于11g,linux5.5做出的测试,单实例数据库做出的测试. 1.注册 Instance到监听器去注册自己的Instance_name与ORACLE_HOME,还可以选择添加global_dbna ...
 - vuex所有核心概念完整解析State Getters  Mutations  Actions
		
vuex是解决vue组件和组件件相互通信而存在的,vue理解起来稍微复杂但一旦看懂择即为好用 安装: npm install --save vuex 引入 import Vuex from 'vuex ...
 - 2.基础:Vue组件的核心概念
		
一.组件基础和注册 组件概念 组件系统是 Vue 的另一个重要概念,他的核心就是封装和复用. 细节 组件的name必须是全局唯一. 二.属性.事件和插槽 组件的三大核心概念:属性.事件和插槽. 属性, ...
 - 《JavaScript核心概念》基础部分重点摘录
		
注:<JavaScript核心概念>适合深入了解JavaScript,比我买的<JavaScript框架设计>语言和内容丰富多了(可能是我水平尚浅吧). 1. 作用域 var ...
 - Javascript本质第一篇:核心概念
		
很多人在使用Javascript之前都至少使用过C++.C#或Java,面向对象的编程思想已经根深蒂固,恰好Javascript在语法上借鉴了Java,虽然方便了Javascript的入门,但要深入理 ...
 - JS核心概念
		
Javascript本质第一篇:核心概念 很多人在使用Javascript之前都至少使用过C++.C#或Java,面向对象的编程思想已经根深蒂固,恰好Javascript在语法上借鉴了Java,虽 ...
 
随机推荐
- C# 两个具有相同属性的类赋值
			
最近有遇到两个类之间的赋值问题,两个类的属性几乎都一样的,所以写了个通过反射获取属性的然后赋值的方法,把一个类的属性的值赋值给另一个类. 框架是.net 4.5 public static D Map ...
 - Zabbix错误”zbx_mem_malloc(): out of memory”解决方法
			
Zabbix Server突然挂了,查看log报错如下: using configuration file: /etc/zabbix/zabbix_server.conf ... [file:dbco ...
 - Netty源码分析之Reactor线程模型详解
			
上一篇文章,分析了Netty服务端启动的初始化过程,今天我们来分析一下Netty中的Reactor线程模型 在分析源码之前,我们先分析,哪些地方用到了EventLoop? NioServerSocke ...
 - [hdu7074]Little prince and the garden of roses
			
显然每种颜色的花是独立的,可以分别求出答案后取$\max$ 对于某种颜色$C$,建立一张二分图,左右分别为$n$行$n$列,且第$i$行和第$j$列有边当且仅当$c_{i,j}=C$ 此时,问题即对边 ...
 - 实验四 Web服务器1-socket编程
			
一.任务详情基于华为鲲鹏云服务器CentOS中(或Ubuntu),使用Linux Socket实现: 1. time服务器的客户端服务器,提交程序运行截图 2. echo服务器的客户端服务器,提交程序 ...
 - [APIO2020]有趣的旅途
			
注意到第一个点是可以钦定的. 那么我们考虑在重心的子树里反复横跳. 每次选择不同子树里的深度最大的点. 在同一颗子树里可能会在lca处出现问题. 那么我们选择重心,要考虑到会不会出现一颗子树不够选的操 ...
 - 洛谷 P7516 - [省选联考 2021 A/B 卷] 图函数(Floyd)
			
洛谷题面传送门 一道需要发现一些简单的性质的中档题(不过可能这道题放在省选 D1T3 中偏简单了?) u1s1 现在已经是 \(1\text{s}\) \(10^9\) 的时代了吗?落伍了落伍了/ ...
 - Codeforces 1368H - Breadboard Capacity(最小割+线段树维护矩阵乘法)
			
Easy version:Codeforces 题面传送门 & 洛谷题面传送门 Hard version:Codeforces 题面传送门 & 洛谷题面传送门 首先看到这种从某一种颜色 ...
 - NOIP 2020 游记
			
第一次写比赛游记,请多多指教! I. 考前 由于最近参加了太多太多比赛了,所以没有敲模板题: 考前一周:主要是在做 AtCoder 的题和 xjoi 的模拟赛,相当于恶补了一些套路吧! 考前一天:上午 ...
 - vector.erase();vector.clear();map.erase();
			
vector::erase()返回下一个iter: STL中的源码: //清除[first, last)中的所有元素 iterator erase(iterator first, iterator l ...