使用git rebase来消除提交历史分叉

在很多时候,随着开发团队的扩容,大家提交代码一段时间过后,就产生很多的合并(merge)节点,对于有代码洁癖的人来说,这个总会觉得很不爽。

master分支混乱的提交历史

那么,merge是如何产生的呢?下面我挑一个常见的冲突的场景,示例一下:

用merge来解决冲突

我们来在master分支模拟一次merge的情况,针对README.md,首先A用户编写了首行代码,并提交到了git服务器上内容如下:

# A用户编写首行代码

同时,B用户在本地编写了首行代码,内容如下:

# B用户编写首行代码

B用户只是在本地电脑上进行了git commit,如下:

$ git commit -am "B用户编写首行代码"
[master c8bfe4f] B用户编写首行代码
 1 file changed, 1 insertion(+), 1 deletion(-)

接下来,B用户使用git pull命令将git服务器代码下载并合并到本地,可以看到产生了冲突:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git pull
remote: Counting objects: 3, done.
remote: Compressing objects: 100% (2/2), done.
remote: Total 3 (delta 0), reused 0 (delta 0)
Unpacking objects: 100% (3/3), done.
From gitee.com:yyxz/test
   45bc85e..fc2c29a  master     -> origin/master
Auto-merging README.md
CONFLICT (content): Merge conflict in README.md
Automatic merge failed; fix conflicts and then commit the result.

接下来,B用户,修改有冲突的README.md文件,解决冲突,如下:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|MERGING)
$ git status
On branch master
Your branch and 'origin/master' have diverged,
and have 1 and 1 different commits each, respectively.
  (use "git pull" to merge the remote branch into yours)
You have unmerged paths.
  (fix conflicts and run "git commit")
  (use "git merge --abort" to abort the merge)

Unmerged paths:
  (use "git add <file>..." to mark resolution)

        both modified:   README.md

no changes added to commit (use "git add" and/or "git commit -a")

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|MERGING)
$ git add README.md

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|MERGING)
$ git commit -m "合并A和B用户编写的首行代码"
[master 5aa451d] 合并A和B用户编写的首行代码

修改后的README.md文件内容如下:

# 合并A和B用户编写的首行代码

现在B将代码提交到git服务器上,可以看到有两个commit(一个是B对首行代码的提交:c8bfe4f,一个是B刚才的merge动作),如下:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git status
On branch master
Your branch is ahead of 'origin/master' by 2 commits.
  (use "git push" to publish your local commits)
nothing to commit, working tree clean

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git push
Counting objects: 6, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (4/4), done.
Writing objects: 100% (6/6), 695 bytes | 0 bytes/s, done.
Total 6 (delta 0), reused 0 (delta 0)
To gitee.com:yyxz/test.git
   fc2c29a..5aa451d  master -> master

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git log --oneline --graph
*   5aa451d (HEAD -> master, origin/master, origin/HEAD) 合并A和B用户编写的首行代码
|\
| * fc2c29a A用户编写首行代码
* | c8bfe4f B用户编写首行代码
|/
* 45bc85e 初始状态

可以看到,提交历史产生了分叉,接下来我们看看如何使用git rebase消除分叉:

git rebase消除分叉

A再次修改首行代码,并提交至git服务器,README.md内容如下:

A再次修改首行代码

B再次在本地修改首行代码,并提交至本地,README.md内容如下:

B再次修改首行代码

commit如下:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git commit -am "B再次修改首行代码"
[master 21a6c2a] B再次修改首行代码
 1 file changed, 1 insertion(+), 1 deletion(-)

接下来,使用git pull --rabase,可以看到冲突,并且进入了rebase状态:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git pull --rebase
remote: Counting objects: 3, done.
remote: Compressing objects: 100% (2/2), done.
remote: Total 3 (delta 0), reused 0 (delta 0)
Unpacking objects: 100% (3/3), done.
From gitee.com:yyxz/test
   5aa451d..ccbb4ff  master     -> origin/master
First, rewinding head to replay your work on top of it...
Applying: B再次修改首行代码
error: Failed to merge in the changes.
Using index info to reconstruct a base tree...
M       README.md
Falling back to patching base and 3-way merge...
Auto-merging README.md
CONFLICT (content): Merge conflict in README.md
Patch failed at 0001 B再次修改首行代码
The copy of the patch that failed is found in: .git/rebase-apply/patch

When you have resolved this problem, run "git rebase --continue".
If you prefer to skip this patch, run "git rebase --skip" instead.
To check out the original branch and stop rebasing, run "git rebase --abort".


bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|REBASE 1/1)
$

此时,B在本地打开README.md文件,可以看到冲突内容如下:

<<<<<<< HEAD
A再次修改首行代码
=======
B再次修改首行代码
>>>>>>> B再次修改首行代码

解决冲突:修改README.md文件,内容如下:

再次合并A和B修改的首行代码

使用git add README.md标注冲突已解决:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|REBASE 1/1)
$ git add README.md

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master|REBASE 1/1)
$ git rebase --continue
Applying: B再次修改首行代码

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git log --oneline --graph
* 1e4c421 (HEAD -> master) B再次修改首行代码
* ccbb4ff (origin/master, origin/HEAD) A再次修改首行代码
*   5aa451d 合并A和B用户编写的首行代码
|\
| * fc2c29a A用户编写首行代码
* | c8bfe4f B用户编写首行代码
|/
* 45bc85e 初始状态

可以看到B用户第二次修改是在A修改之后,但是B的提交commit编号21a6c2a被新的编号1e4c421所取代,commit的内容不变。

这时我们看到HEAD -> master,origin/master, origin/HEAD不一致。是因为B用户在本地的修改还未提交至git服务器,通过git status也可以看出来,有1 commit未提交至git服务器:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git status
On branch master
Your branch is ahead of 'origin/master' by 1 commit.
  (use "git push" to publish your local commits)
nothing to commit, working tree clean

提交后,就OK了。如下:

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git push
Counting objects: 3, done.
Delta compression using up to 4 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (3/3), 355 bytes | 0 bytes/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To gitee.com:yyxz/test.git
   ccbb4ff..1e4c421  master -> master

bianxh@DESKTOP-BIANXH MINGW64 /d/Workspaces/gitee/test (master)
$ git log --oneline --graph
* 1e4c421 (HEAD -> master, origin/master, origin/HEAD) B再次修改首行代码
* ccbb4ff A再次修改首行代码
*   5aa451d 合并A和B用户编写的首行代码
|\
| * fc2c29a A用户编写首行代码
* | c8bfe4f B用户编写首行代码
|/
* 45bc85e 初始状态

现在我们登录到服务器看下最新的内容,如下:

使用rebase解决冲突后的README.md文件内容

可以通过项目网络图直观地看到,如下:


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

推荐阅读更多精彩内容