git 分支管理

下一篇传送门:git 代码合并

Git

观察过很多使用git的人,只会用add, commit,push,pull这几个命令,包括恢复版本之类的,多半也会暴力的删除整个项目,再重新clone干净的代码。虽说也能工作了,但是无疑没有领会到git的精华。
这篇文章主要说明如何使用分支使我们的开发工作更加顺滑,如何让分支成为你日常工作流不可缺失的一部分。

git branch 用法##

git branch   //列出所有的分支
git branch <branch> //创建名为<branch>的分支,但是不会切换过去
git branch -d <branch>  //删除指定分支,这是一个“安全”操作,git会阻止你删除包含未合并更改的分支。
git branch -D <branch>  //强制删除分支
git branch -m <branch> //重新命名当前分支

在日常开发中,无论是修复一个bug或者添加一个功能,我们都应该新建一个分支来封装我们的修改。这样可以保证我们不稳定的代码永远不会提交到主代码分支上。

下面我们具体来看在执行分支有关的操作,分支的变化:

创建分支##

分支只是指向提交的指针,理解这一点很重要,当你创建新分支,实际上只是创建了一个新的指针,仓库本身不会受到影响,一开始你的仓库只有一条分支:

master

然后你执行下面的命令创建一个分支,用于加一个新feature:

git branch new-feature

new-feature

当然执行后,你只是创建了这个分支,还需要执行git chekcout new-feature切换到new-feature分支,然后再使用git add,git commit

删除分支##

假如你已经开发完了new-feature,并且已经commit代码了, 你就可以自由删除这个分支了。

git branch -d new-feature

如果分支还没有合入master,会报下面的错误:

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

这时候你可以合并分支(下面会说如何合并分支),如果你真的确定了要删除分支,可以用-D执行强制删除:

git branch -D new-feature

切换分支(git checkout)##

git checkout 命令允许你切换到用git branch创建的分支。切换分支会更新当前工作目录中的文件,还可以用git log查看当前分支的历史。
用法

git checkout <existing-branch> //切换到一个已有分支上
git checkout -b <new-branch> // -b 标记 可以方便让你先创建一个新的new-branch,再直接切换过去
git checkout -b <new-branch> <existing-branch> //在已有的分支上创建分支,原来的分支使新分支的基

git branchgit chekout 是一对好基友,你可以使用git checkout在不同的功能分支或者bug分支之间切换,而不产生相互影响。

关于分离的HEAD(detached HEAD)
一般我们有时候需要恢复到以前commit的版本上查看原来的一些文件时,我们会git checkout commit的hash码或者tag, 这时候会提醒我们进入了detached HEAD

detached HEAD

拿上图举例,我们当前的HEAD在4,然后假设 git checkout 2,我们的代码回到2的状态了,通常git会显示下面的warning:

You are in 'detached HEAD' state. You can look around, make experimental
changes and commit them, and you can discard any commits you make in this
state without impacting any branches by performing another checkout.

If you want to create a new branch to retain commits you create, you may
do so (now or later) by using -b with the checkout command again. Example:

  git checkout -b new_branch_name

HEAD is now at 2

如果在这种状态下开发,然后又add,commit,没有分支可以让你回到之前的状态。当你不可避免的需要checkout到另外一个分支,想再回来就是不可能的了,因为像图中的那个X状态,根本就不在分支上,你将再也不能引用你之前添加的代码了。

重点:永远记得在开发分支上开发,不要在分离的HEAD上开发,这可以确保你可以引用到你的新提交,如果只是checkout到以前看看无所谓,如果真的需要在以前的版本上添加什么代码,记得上面warning 中的git checkout -b new_branch_name,是自己处于一个确切的分支中。

远离detached HEAD,我们来看实际上git分支流程是什么样子的:

git branch new-feature
git checkout new-feature

接下来我们做一些代码改动,提交:

git add <some file>
git commit -m "A new feature"
##你接着改代码,接着提交很多次

当你git log会显示每一次的commit, 和master分支完全独立,当你checkout到master分支去,再git log,会发现new-feature分支的提交都不在,这就是不影响master分支。
这时候,你可以考虑合并new-feature或者在master分支上开始别的工作。

合并(git merge)##

合并是git将被fork的历史放回到一起的方式。 git merge 命令允许你将 git branch 创建的多条分支合并成一个。
用法

git merge <branch>  //将指定分支并入当前分支
git merge --no-ff <branch>  //将指定分支并入当前分支,但 总是 生成一个合并提交(即使是快速向前合并)。这可以用来记录仓库中发生的所有合并。

一旦在新分支上完成开发,我们需要把新分支的提交合并到主分支,git会根据目前分支之间的结构信息,选择不同的算法来完成合并:

  • 快速向前合并
  • 三路合并

快速向前合并
当new-feature的分支与原有的master分支呈现线性关系时,执行快速向前合并,git将当前的HEAD指针快速移到目标分支的顶端,master分支也就具有了new-feature分支的历史了,如图:

快速向前合并

来看一个快速向前合并的实例:

# 开始新功能
git checkout -b new-feature master

# 编辑文件
git add <file>
git commit -m "开始新功能"

# 编辑文件
git add <file>
git commit -m "完成功能"

# 合并new-feature分支
git checkout master
git merge new-feature
git branch -d new-feature

对于合作开发的人少的项目,这是一种主要的工作流,合作开发的人多的话,主master分支经常都会有新提交,如果你的new-feature耗时比较久,再提交时,master分支可能已经过去几个版本了,这时候就需要下面的三路合并了。

三路合并
但是如果master分支在new-feature分离后,又有了新的提交,即开始分叉了,git只能执行三路合并,三路合并使用一个专门的提交来合并两个分支的历史。

已经分叉的branch

所谓的三路也就是:两个分支的顶端以及他们共同的祖先。 在执行三路合并后:

三路合并后

使用三路合并产生的合并提交作为两个分支的连接标志。

解决冲突
如果两个分支对同一个文件的同一部分均有修改时,git将无法判断应该使用哪个,这时候合并提交会停止,需要你手动解决这些冲突。你可以使用git status来查看哪里存在冲突,很多时候我都会在目录下执行grep -rn HEAD来查看哪些文件里有这个标记,有这个标记的地方都是有冲突的。

当修改完所有的冲突后,git add所有的冲突文件,运行git commit生成一个合并提交,这和提交一个普通快照的流程相同。提交冲突只会存在在三路合并中,快速向前合并中不可能出现针对同一文件同一部分的不一样修改。

下面实例来看看三路合并是怎么产生的:

# 开始新功能
git checkout -b new-feature master

# 编辑文件
git add <file>
git commit -m "开始新功能"

# 编辑文件
git add <file>
git commit -m "完成功能"

# 在master分支上开发
git checkout master

# 编辑文件
git add <file>
git commit -m "在master上添加了一些极其稳定的功能"

# 合并new-feature分支
git merge new-feature
git branch -d new-feature

这时候,merge会停止,因为无法将 master 直接移动到 new-feature。所以需要你手动合并冲突后再提交。

git merge会产生合并提交,有的人会选择使用git rebase来合并以确保一个干净的提交历史。关于这两个的区别,我会另写一篇介绍。git 代码合并

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

推荐阅读更多精彩内容