分支与分支工作流

在项目的开发过程中,分支与合并是很常见的操作。以我曾经管理过的一个长期的项目为例,由于项目的总体需求规模比较庞大,全部开发完成需要很长的时间,而且随着业务的增长后续不断会有需求的边界也会逐步的扩张。为了规避风险因此我们采用敏捷的开发模式,同时将需求按优先级做迭代增量开发和交付,持续和尽早获得反馈。短期内(3个月)项目就上线了,随后每个月都会发布一个包含了新需求的版本。在这中间可能也会有修复紧急feature、issue的临时发布。

在上面提到的项目实例,假设新的需求开发、生产环境紧急的bug修复工作等都是直接在master分支(Git在新建一个仓库时,默认只有一个名为master的分支)内进行,会发生什么样的后果?


图-1

(注:图-1 带箭头实线表示的不是提交方向、时间方向,而是指针,指向当前提交的上一次提交(parent))

实例中最容易发生的后果之一,如图-1所示,master最近一次的提交C2是发布到生产环境的版本。随后开发人员提交了包含了新特性的代码C3以及C4开始做集成测试。此时在生产环境发现了一个bug需要紧急修复并发布。但是现在master内的代码已经包含了C3等不能发布到生产环境的代码提交。这种情况下要么只有等C3测试通过可以发布到生产环境时,和bugfix一起发布;又或者将版本回撤到C2,修复bug后将重新提交C3和C4。两种方案都是非常差的选择。

为了规避这类问题,就需要使用分支功能将新工作从开发的主线上分离开来,以免影响开发主线。例如生产环境单独使用master分支,其它的工作如新需求开发、紧急修复等分别使用不同的分支,在工作完成后需要发布时再和并回master。可以根据项目的实际情况来制定分支策略和发布管理的工作流程。

1. 分支的创建

在git中创建一个分支很简单:git branch <newbranchname>。在上面的例子中,使用分支的简单方案就是,在C2阶段,将开发C3等特性的工作放入新的分支去做。创建一个分支:

$ git branch feature-c3

检出分支:

git checkout feature-c3

检出分支时会将该分支的最后一次提交检出到暂存区和工作区。另外,也可以使用git checkout -b feature-c3命令来创建并立即检出这个新分支,这条命令等效于先后执行上面的两条命令。

接下来我们来模仿在feature-c3分支中增加新的特性:

  • 修改README.md文件,新增一行内容并保存:This line is demo for new feature c3.
  • 新增一个featurec4.txt,文件内容保存一行文字:This line is demo for new feature c4.
    然后分别使用add、commit将修改提交到版本库。

此时C2版本发布到生产环境的系统发现了严重的bug #07,需要紧急修复和发布。目前C3和C4都是在feature-c3上提交的,master仍然是c2的源码版本。所以我们需要使用master分支的版本来修复bug并发布。

首先运行git checkout master命令检出master分支。
然后创建一个基于master的hotfix分支git checkout -b hotfix0.1_07,然后修改README.md文件,在第6行保存段文字“This is a hotfix for v0.1.0”,然后add、commit文件。

2. 分支的合并

假设上面修改了README.md文件后就修复了bug #07并通过了验收测试,现在要将hotfix分支的修改合并回master并发布到生产环境。
首先要将分支切换回master:git checkout master。现在如果查看README.md文件会发现内容仍然只有5行(C2时的版本)。然后运行merge命令git merge hotfix0.1

$ git merge hotfix0.1
Updating 4a567b9..fa7f53c
Fast-forward
 README.md | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

打开gitsample工作区可以看到README.md文件已经发生了改变,hotfix分支的修改已经合并进来了。
现在hotfix分支已经完成了使命,我们可以使用git branch -d hotfix0.1命令将该分支删除。

再来查看仓库里的分支信息:git branch

$ git branch
  feature-c3
* master

从命令返回结果可以看到,仓库里只剩下feature-c3和master两个分支。master左侧有一个*,表示这是当前检出的分支。

feature-c3分支的内容还没有被合并到master,此时我们来试着删除这个分支:

$ git branch -d feature-c3
error: The branch 'feature-c3' is not fully merged.
If you are sure you want to delete it, run 'git branch -D feature-c3'.

命令返回错误信息:c3分支的内容还没有完全合并,不能删除。现在假设feature分支的开发已经完成并通过测试验收,可以发布到生产环境了。让我们再次合并分支:

$ git merge feature-c3
Auto-merging README.md
CONFLICT (content): Merge conflict in README.md
Automatic merge failed; fix conflicts and then commit the result.

命令提示README.md文件的内容冲突,自动合并失败。需要人工干预重复然后再提交。
我们先用git log -1看下最后一次的提交日志:

efrey@EKStudio MINGW64 ~/Desktop/gitsample (master|MERGING)
$ git log
commit fa7f53c462f5500d00f2968b7f7ab42d307cdbc0 (HEAD -> master)
Author: Efrey Kong <efreykong@outlook.com>
Date:   Tue Aug 20 10:15:56 2019 +0800

    hotfix 0.1.0

可以看到Git bash命令行提示符上方的显示内容由

efrey@EKStudio MINGW64 ~/Desktop/gitsample (master)

变成了

efrey@EKStudio MINGW64 ~/Desktop/gitsample (master|MERGING)

并且日志中最后一次的提交仍然是合并hotfix分支操作。
接下来我们再打开README.md看下文件冲突的内容:

# gitsample

This line is the first time modified.

This line is the second time modified.
<<<<<<< HEAD
This is a hotfix for v0.1.0
=======
This line is a demo for feature-c3.
>>>>>>> feature-c3

<<<<<<< HEAD=======中间的内容是当前master分支的内容;
======= >>>>>>> feature-c3中间的内容是feature-c3分支的内容。

原来在hotfix分支中,我们在README.md文件的第6行增加了一段“This is a hotfix for v0.1.0”。
而在feature-c3分支中,同样的位置增加了一段“This line is a demo for feature-c3.”。我们来处理下这个冲突:

# gitsample

This line is the first time modified.

This line is the second time modified.
This is a hotfix for v0.1.0
This line is a demo for feature-c3.

保存修改,然后分别运行git add .git commit -m "feature-c3"提交。此时再来查看最后一次提交日志:

efrey@EKStudio-ZBook MINGW64 ~/Desktop/gitsample (master)
$ git log -1
commit 23b2190e6821953f8bba6935faef56cf3532b694 (HEAD -> master)
Merge: fa7f53c 49418b5
Author: Efrey Kong <efreykong@outlook.com>
Date:   Tue Aug 20 10:55:25 2019 +0800

    feature-c3

合并成功了。此时删除feature-c3:

$ git branch -d feature-c3
Deleted branch feature-c3 (was 49418b5).

3. 分支工作区内容的暂存

4. 分支工作流程模型

4.1. develop-master模型

A successful Git branching model一文提到了一种比较实用的Git工作流程模型,也被称为develop-master模型。

A successful Git branching model

在这种模型中,git中存在两个长期分支:master和develop。其它的分支如hotfix、feature、release等属于辅助分支,按需要创建,在工作完成后合并回长期分支并删除辅助分支。

master作为系统稳定版的主干,最近一次提交是永远和生产环境保持一致的。dev分支在项目的一开始就和master分离,作为持续开发的主干。

当生产环境系统发现了bug,则在master的主干上创建一个hotfix的分支用于修复bug。bug修复测试通过后,再将这个hotfix分支合并回master,同时dev也会吸收hotfix分支中的修复。合并后删除hotfix分支。

在开发下一次发布的主要需求功能,或者要开发新的特性时,则从dev分支创建新的分支,开发完成后合并回dev分支,并删除这个新的分支。当dev分支中的可以发布时,从dev创建release分支,release分支只接受bug fix的提交(bug fix的提交也需要持续的合并回dev分支)。测试完成后,将release分支合并到master发布版本到生产环境。合并后删除release分支。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352