【CI/CD】Git Flow 分支模型

Git Flow 分支模型

1.前言

Git Flow 模型(本文所阐述的分支模型)构思于 2010 年,也就是 Git 诞生后不久,距今已有 10 多年。在这 10 多年中,Git Flow 在许多软件团队中大受欢迎。

在这 10 多年里,Git 本身已经风靡全球,而使用 Git 开发的最流行的软件类型也更多地转向了网络应用。网络应用通常是持续交付的,不会回滚,也不需要支持多个版本的软件同时运行。这与原作者在 10 年前写这篇博文时所考虑的软件类型不同。

如果你的团队正在进行软件的持续交付,建议采用更简单的工作流程(比如 GitHub Flow),而不是试图把 Git Flow 强塞进你的团队。不过,如果您正在构建明确版本化的软件,或者您需要支持软件的多个版本,那么 Git Flow 可能仍然适合您的团队。

最后,请记住,一劳永逸式的通用解决方案是不存在的。考虑一下你自己的项目背景,再做决定。

在这里插入图片描述

2.Main branches 主要分支

在核心部分,开发模型受到了现有模型的极大启发。中心仓库拥有两个无限生命周期的主分支:

  • master
  • develop

在这里插入图片描述
每个 Git 用户对位于 origin 的 master 分支都不会陌生。与 master 分支平行的另一个分支叫 develop 分支。

我们把 origin/master 视为主分支,在这里,HEAD 的源代码总是反映生产就绪的状态。

我们认为 origin/develop 是主分支,HEAD 的源代码总是反映下一个版本最新交付的开发变更状态。有人称其为 “集成分支”。每晚的自动构建都是从这里开始的。

develop 分支中的源代码达到稳定点并准备发布时,所有变更都应以某种方式合并回 master 分支,然后标记上发布编号。具体做法将在后面讨论。

因此,每次将变更合并回 master 版本时,这就是一个新的生产版本。我们倾向于严格遵守这一点,因此理论上我们可以使用 Git 钩子脚本,在每次 master 版本有提交时自动构建软件并将其发布到生产服务器上。

3.Supporting branches 辅助分支

除了主分支 master 和开发分支 develop,我们的开发模式还使用各种辅助分支来帮助团队成员之间的 并行开发简化功能跟踪为生产发布做准备,以及 协助快速修复实时生产问题。与主分支不同,这些分支的生命周期总是有限的,因为它们最终会被移除。

我们可能使用的不同类型的分支包括

  • Feature branches(特性分支)
  • Release branches(发布分支)
  • Hotfix branches(热修复分支)

每种分支都有其特定的目的,并受严格的规则约束,哪些分支可以作为其原始分支,哪些分支必须作为其合并目标。我们稍后将详细介绍它们。

从技术角度看,这些分支并不 “特殊”。分支类型是根据我们的使用方式来分类的。当然,它们都是普通的 Git 分支。

3.1 Feature branches 特性分支

来自于 develop,必须合并回 develop

在这里插入图片描述
分支命名约定:只要不用 masterdeveloprelease-*hotfix-* ,其他都可以。

feature 分支(有时也称作主题分支 Topic branches)用于 为即将发布或未来发布的版本开发新特性。在开始开发某个功能时,可能还不知道该功能将被集成到哪个目标版本中。feature 分支的本质是,只要该特性还在开发阶段,它就一直存在,但最终会被合并回 develop(将新特性添加到即将发布的版本中)。

特性分支通常只存在于开发者版本库中,而不存在于原始版本库(origin)中

(1)创建 feature 分支

开始开发新功能时,从 develop 分支创建分支。

$ git checkout -b myfeature develop
Switched to a new branch "myfeature"

(2)将已完成的功能并入 develop 分支

已完成的功能可以合并到 develop 分支,以便在即将发布的版本中加入这些功能:

$ git checkout develop
Switched to branch 'develop'
$ git merge --no-ff myfeature
Updating ea1b82a..05e9557
(Summary of changes)
$ git branch -d myfeature
Deleted branch myfeature (was 05e9557).
$ git push origin develop

使用 - - no - ff 标志后,即使合并可以通过快进执行,也会始终创建一个新的提交对象。这样可以避免丢失 feature 分支历史存在的信息,并将共同添加该特性的所有提交组合在一起。

在这里插入图片描述
在后一种情况下,我们无法从 Git 历史记录中看到哪些提交对象一起实现了某个特性,而必须手动阅读所有日志信息。在后一种情况下,还原整个特性(即一组提交)确实是个令人头疼的问题,而如果使用 - - no - ff 标志,就很容易做到。

是的,这会多创建几个(空)提交对象,但收益远大于代价。

3.2 Release branches 发布分支

来自于 develop,必须合并回 developmaster

分支命名约定:release-*

release 分支支持新生产版本的准备工作。release 分支允许在最后一刻进行校验。此外,它们还可以进行小的错误修复,并为发布版本准备元数据(版本号、构建日期等)。在 release 分支上完成所有这些工作后,develop 分支就可以接收下一个大版本的功能了。

develop 分支分离出新版本分支的关键时刻是 develop 分支(几乎)反映了新版本的理想状态。至少所有针对即将发布的版本的功能都必须在此时合并到 develop 分支。而针对未来版本的所有功能则不一定,它们必须等到 release 分支分支化之后。

正是在 release 分支开始时,即将发布的版本才会被分配一个版本号,而不是更早。在此之前,develop 分支反映的是 “下一个版本” 的变更,但在 release 分支启动之前,“下一个版本” 最终是 0.3 0.3 0.3 还是 1.0 1.0 1.0 并不清楚。这个决定是在 release 分支启动时做出的,并由项目的版本号递增规则来执行。

(1)创建 release 分支

release 分支是从 develop 分支创建的。例如, 1.1.5 1.1.5 1.1.5 版是当前的生产版本,而我们即将发布一个大版本。develop 分支已经为 “下一个版本” 做好了准备,我们决定将其命名为 1.2 1.2 1.2 版(而不是 1.1.6 1.1.6 1.1.6 版或 2.0 2.0 2.0 版)。因此,我们建立了分支,并给 release 分支起了一个反映新版本号的名字:

$ git checkout -b release-1.2 develop
Switched to a new branch "release-1.2"
$ ./bump-version.sh 1.2
Files modified successfully, version bumped to 1.2.
$ git commit -a -m "Bumped version number to 1.2"
[release-1.2 74d9424] Bumped version number to 1.2
1 files changed, 1 insertions(+), 1 deletions(-)

创建新分支并切换到该分支后,我们就会提升版本号。在这里,bump-version.sh 是一个虚构的 shell 脚本,它会更改工作副本中的某些文件,以反映新版本。(当然,这也可以是手动更改,关键是某些文件会发生变化)。然后,提交被修改的版本号。

这个新分支可能会存在一段时间,直到正式发布。在此期间,错误修复可能会应用到该分支(而不是 develop 分支)。严禁在此添加大型新功能。它们必须合并到 develop 分支,因此要等到下一个大版本发布。

(2)完成 release 分支

release 分支的状态准备好成为真正的发布版本时,需要执行一些操作。首先,将 release 分支合并到 master 分支(因为 master 分支上的每次提交都是一次新的发布,切记)。其次,必须对 master 分支上的提交进行标记,以方便将来引用这个历史版本。最后,需要将 release 分支上的修改合并回 develop 分支,以便未来的版本也包含这些错误修复

Git 的前两个步骤:

$ git checkout master
Switched to branch 'master'
$ git merge --no-ff release-1.2
Merge made by recursive.
(Summary of changes)
$ git tag -a 1.2

现在,发布工作已经完成,并打上了标签,以备将来参考。(你也可以使用 -s-u <key> 标记对标签进行加密签名。)

为了保留 release 分支中的改动,我们需要把它们合并回 develop 分支。在 Git 中:

$ git checkout develop
Switched to branch 'develop'
$ git merge --no-ff release-1.2
Merge made by recursive.
(Summary of changes)

这一步很可能导致合并冲突(甚至可能,因为我们已经更改了版本号)。如果是这样,请修复并提交。

现在我们真的完成了,release 分支也可以删除了,因为我们不再需要它了:

$ git branch -d release-1.2
Deleted branch release-1.2 (was ff452fe).

3.3 Hotfix branches 热修复分支

来自于 master,必须合并回 developmaster

分支命名约定:hotfix-*

在这里插入图片描述
hotfix 分支与 release 分支非常相似,都是为新的生产版本做准备,尽管是计划外的。它们产生的原因是,必须立即对实时生产版本中不希望出现的状态采取行动。当必须立即解决生产版本中的关键错误时,可以从标记生产版本的 master 分支上的相应标记中分出一个 hotfix 分支。

这样做的好处是,团队成员(在开发分支上)的工作可以继续进行,而另一个人则在准备快速的生产修复

(1)创建 hotfix 分支

hotfix 分支是从 master 分支创建的。例如, 1.2 1.2 1.2 版是当前正在运行的生产版本,由于一个严重的错误而造成了很多麻烦。但开发中的更改还不稳定。这时,我们可以分离出一个 hotfix 分支,开始修复 1.2 1.2 1.2 版本的错误。

$ git checkout -b hotfix-1.2.1 master
Switched to a new branch "hotfix-1.2.1"
$ ./bump-version.sh 1.2.1
Files modified successfully, version bumped to 1.2.1.
$ git commit -a -m "Bumped version number to 1.2.1"
[hotfix-1.2.1 41e61bb] Bumped version number to 1.2.1
1 files changed, 1 insertions(+), 1 deletions(-)

分支后不要忘记提升版本号!

然后,修复错误并在一个或多个单独的提交中提交修复。

$ git commit -m "Fixed severe production problem"
[hotfix-1.2.1 abbe5d6] Fixed severe production problem
5 files changed, 32 insertions(+), 17 deletions(-)

(2)完成 hotfix 分支

完成后,需要将漏洞修复合并回 master 分支,但同时也需要合并回 develop 分支,以确保下一个版本也包含该漏洞修复。这与 release 分支的完成方式完全类似。

首先,更新主版本并标记发布版本。

$ git checkout master
Switched to branch 'master'
$ git merge --no-ff hotfix-1.2.1
Merge made by recursive.
(Summary of changes)
$ git tag -a 1.2.1

可以使用 -s-u <key> 标记对标签进行加密签名。

接下来,在 develop 中也包含错误修正:

$ git checkout develop
Switched to branch 'develop'
$ git merge --no-ff hotfix-1.2.1
Merge made by recursive.
(Summary of changes)

这里的规则有一个例外,那就是当当前存在 release 版分支时,热修复程序的更改需要合并到 release 分支,而不是 develop 分支。将错误修正反向合并到 release 分支,最终会导致在 release 分支完成后,错误修正也被合并到 develop 分支。(如果 develop 中的工作立即需要这个错误修正,而又不能等到 release 分支完成,那么现在也可以安全地将错误修正合并到 develop 中)。

最后,删除临时分支:

$ git branch -d hotfix-1.2.1
Deleted branch hotfix-1.2.1 (was abbe5d6).

4.总结

虽然这个分支模型并没有什么令人震惊的新内容,但这篇文章开头提到的 “全局” 图在我们的项目中却非常有用。它形成了一个优雅的心智模型,易于理解,并能让团队成员对分支和发布流程形成共识。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/29726.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

最大子数组和【力扣53】

一、解题思路 Max[i]表示&#xff1a;以nums[i]为开头的所有连续子数组和的最大值。 由此可以推出Max[i-1]和Max[i]的关系&#xff1a; 若Max[i]>0&#xff1a;Max[i-1]nums[i-1]Max[i]&#xff1b; 否则&#xff1a;Max[i-1]nums[i-1]&#xff1b; 则ansMAX&#xff0…

java try-with-resources自动关闭资源

try-with-resources 是 Java 7 引入的一种语法&#xff0c;用于自动关闭实现了 AutoCloseable 接口的资源。它可以简化代码中关闭资源的操作&#xff0c;确保资源得到正确释放&#xff0c;避免内存泄漏。 使用 try-with-resources 的语法如下&#xff1a; try (资源的声明和初…

RISC-V走向开放服务器规范

原文&#xff1a;RISC-V Moving Toward Open Server Specification 作者&#xff1a;Agam Shah 转载自&#xff1a;https://www.hpcwire.com/2023/07/24/risc-v-moving-toward-open-server-specification/ 中文翻译&#xff1a; 2023年7月24日 RISC-V International目前正…

亚马逊 EC2服务器下部署java环境

1. jdk 1.8 安装 1.1 下载jdk包 官网 Java Downloads | Oracle tar.gz 包 下载下来 1.2 本地连接 服务器 我用的是亚马逊的ec2 系统是 ubuntu 的 ssh工具是 Mobaxterm , 公有dns 创建实例时的秘钥 链接 Mobaxterm 因为使用的 ubuntu 所以登录的 名称 就是 ubuntu 然后 …

ts项目中引入js包,如vue-simple-uploader

有些包如vue-simple-uploader&#xff0c;使用npm下载后&#xff0c;npm仓库中没有对应的types类型声明&#xff0c;会报错&#xff0c;没有找到对应的types类型文件。 解决&#xff1a; 在项目根目录下&#xff0c;创建自定义的类型声明文件&#xff0c;如&#xff0c;self-…

【Ubuntu】简化反向代理和个性化标签页体验

本文将介绍如何使用Docker部署Nginx Proxy Manager和OneNav&#xff0c;两个功能强大且易用的工具。Nginx Proxy Manager用于简化和管理Nginx反向代理服务器的配置&#xff0c;而OneNav则提供个性化的新标签页体验和导航功能。通过本文的指导&#xff0c;您将学习如何安装和配置…

vue组件通信的方式?

父子通信&#xff1a; 1、父传子&#xff1a;在父组件的子组件标签绑定一个自定义属性&#xff0c;子组件通过props获取父组件传递的数据。 //父 <child :data"list" del"idx > list.splice(idx, 1)" ref"child" :isShow.sync"isS…

Redis可以用作数据库吗?它的适用场景是什么?

是的&#xff0c;Redis可以用作数据库。虽然Redis通常被认为是一个内存数据库&#xff08;in-memory database&#xff09;&#xff0c;但它也可以通过持久化机制将数据保存在磁盘上&#xff0c;以便在重启后恢复数据。 Redis的适用场景包括但不限于以下几个方面&#xff1a; …

程序使用Microsoft.XMLHTTP对象请求https时出错解决

程序中使用Microsoft.XMLHTTP组件请求https时出现如下错误&#xff1a; 出错程序代码示例&#xff1a; strUrl "https://www.xxx.com/xxx.asp?id11" dim objXmlHttp set objXmlHttp Server.CreateObject("Microsoft.XMLHTTP") objXmlHttp.open "…

电脑关机程序

//关机程序 1、电脑运行起来后&#xff0c;1分钟内关机。 2、如果输入&#xff1a;我是猪。就取消关机。 #include<stdio.h> #include<string.h> int main() { char input[20] { 0 }; system("shutdown -s -t 60"); again: printf(&quo…

Cesium相机理解

关于cesium相机&#xff0c;包括里面内部原理网上有很多人讲的都很清楚了&#xff0c;我感觉这两个人写的都挺好得&#xff1a; 相机 Camera | Cesium 入门教程 (syzdev.cn) Cesium中的相机—setView&lookAtTransform_cesium setview_云上飞47636962的博客-CSDN博客上面这…

【Linux】进程间通信——system V共享内存

目录 写在前面的话 System V共享内存原理 System V共享内存的建立 代码实现System V共享内存 创建共享内存shmget() ftok() 删除共享内存shmctl() 挂接共享内存shmat() 取消挂接共享内存shmdt() 整体通信流程的实现 写在前面的话 上一章我们讲了进程间通信的第一种方式…

Linux基础学习

文章目录 Linux命令学习Linux环境准备Linux命令行学习Linux命令行格式与文件系统linux实用命令笔记Linux文件权限查看 Linux命令学习 理解Linux命令是什么 &#xff08;图形化的操作&#xff0c;文件查看&#xff0c;浏览器打开&#xff09; 你打开一个谷歌浏览器&#xff0c;…

微积分入门:总结归纳汇总(一)

基础 标准符号约定: ( s i n x ) n (sinx)^n (sinx)

Spring Aop组成部分

Spring Aop &#xff08;Aspect orirnted Programming&#xff09;面向切面编程,是对面向对象编程的一种补充&#xff0c;是一种编程思想&#xff0c;是对某一类的事情的集中处理。 比如用户登录状态的验证状态&#xff0c;在之前写servlet版本的博客系统&#xff0c;这里就有…

模拟实现消息队列项目(系列4) -- 服务器模块(内存管理)

目录 前言 1. 创建MemoryDataCenter 2. 封装Exchange 和 Queue方法 3. 封装Binding操作 4. 封装Message操作 4.1 封装消息中心集合messageMap 4.2 封装消息与队列的关系集合queueMessageMap的操作 5. 封装未确认消息集合waitMessage的操作 6. 从硬盘中恢复数据到内存中 7. Memo…

ceph pg inconsistent修复(unexpected clone)

问题概述&#xff1a; ceph -s 显示pg 10.17 inconsistent 且命令ceph pg repair 10.17无法修复&#xff0c;/var/log/ceph/cep-osd.3.log报错内容如下&#xff1a; pg 10.17 osd [3,4] 权威副本osd&#xff1a;3 repair 10.17 10:e889b16a:::rbd_data.88033092ad95.00000000…

Linux 命令之 - scp(从远端机器拉取数据)

scp是secure copy的简写&#xff0c;用于在Linux下进行远程拷贝文件的命令&#xff0c;和它类似的命令有cp&#xff0c;不过cp只是在本机进行拷贝不能跨服务器&#xff0c;而且scp传输是加密的。 命令格式&#xff1a; scp [参数] [原路径] [目标路径]从本地服务器复制到远程…

面试攻略,Java 基础面试 100 问(一)

面向对象的特征有哪些方面? 抽象(Encapsulation)&#xff1a;封装的目的是为了保证变量的安全性&#xff0c;使用者不必在意具体实现细节&#xff0c;而只是通过外部接口即可访问类的成员 如果不进行封装&#xff0c;类中的实例变量可以直接查看和修改&#xff0c;可能给整个…