关于Git分支基础知识的一些笔记

躺平是一种积极的生活态度 —–山河已无恙

写在前面

  • 今天和小伙伴们分享一些Git分支的笔记
  • 学习的原因,关于Git分支之前简单看了下,就直接开始玩了,结果整不明白,乱七八糟
  • 看着很糟心,所以觉得有必要系统的学习下
  • 博文为《Pro Git》读书笔记整理,基本上是书里的知识
  • 感谢开源这本书的作者和把这本书翻译为中文的大佬们
  • 理解不足小伙伴帮忙指正,书很不错,感兴趣小伙伴可以去拜读下

躺平是一种积极的生活态度 —–山河已无恙


Git 分支

关于Git分支管理的一些建议,一般可以在本地解决的问题要在本地解决,本地合并(要申请合并到的远程分支),本地解决冲突,如果自己的分支,只顾着开发,不做合并或者变基的操作,在自己的分支上越走越远,慢慢的和远程主主分支差的太远,申请合并一堆冲突…,那是一件很糟糕的事,尤其是对代码的审核者而言,会认为申请合并者是一个不会使用的Git的开发者。
在这里插入图片描述
在学习Git分支之前,我们来了解一些理论知识

分支理论

Git 保存的不是文件变化或者差异,而是一系列不同时刻的快照

在进行提交操作时,Git会保存一个提交对象(commit object)。包括:

  • 一个指向暂存内容快照指针
  • 作者的姓名邮箱
  • 提交时输入的信息
  • 指向它的父对象指针

首次提交产生的提交对象没有父对象普通提交操作产生的提交对象有一个父对象,而由多个分支合并产生的提交对象多个父对象

假设现在有一个工作目录,里面包含了三个将要被暂存(add)和提交(commit)的文件。暂存操作会为每一个文件计算校验和(使用SHA-1哈希算法),然后会把当前版本的文件快照保存到Git仓库中(Git使用blob对象来保存它们),最终将校验和加入到暂存区域等待提交

当使用git commit 进行提交操作时,Git会先计算每一个子目录(本例中只有项目根目录)的校验和,然后在 Git 仓库中这些校验和保存为树对象。随后,Git 便会创建一个提交对象,它除了包含上面提到的那些信息外,还包含指向这个树对象(项目根目录)的指针。如此一来,Git就可以在需要的时候重现此次保存的快照。

现在,Git仓库中有五个对象:三个blob对象(保存着文件快照)一个树对象(记录着目录结构和blob对象索引)以及一个提交对象(包含着指向前述树对象的指针和所有提交信息)

做些修改后再次提交,那么这次产生的提交对象会包含一个指向上次提交对象(父对象)的指针

Git的分支,其实本质上仅仅是指向提交对象的可变指针。Git的默认分支名字是master。在多次提交操作之后,你其实已经有一个指向最后那个提交对象的master分支master分支会在每次提交时自动向前移动。

Git的master分支并不是一个特殊分支。它就跟其它分支完全没有区别。之所以几乎每一个仓库都有master分支,是因为git init命令默认创建它,并且大多数人都懒得去改动

1
2
3
4
5
PS E:\docker\git_example> git init
Initialized empty Git repository in E:/docker/git_example/.git/
PS E:\docker\git_example> git status
On branch master
.........

分支创建

Git 是怎么创建新分支的呢?

很简单,它只是为你创建了一个可以移动的新的指针。在当前所在的提交对象上创建一个指针。比如,创建一个testing分支,你需要使用git branch命令:

1
git branch testing

Git 又是怎么知道当前在哪一个分支上呢?

也很简单,它有一个名为 HEAD 的特殊指针。在Git中,它是一个指针,指向当前所在的本地分支(译注:将HEAD想象为当前分支的别名)。git branch命令仅仅创建一个新分支,并不会自动切换到新分支中去。

你可以简单地使用git log命令查看各个分支当前所指的对象。 提供这一功能的参数是 --decorate

1
2
3
4
5
$ git log --oneline --decorate
f30ab (HEAD -> master, testing) add feature #32 - ability to add new
formats to the central interface
34ac2 Fixed bug #1328 - stack overflow under certain conditions
98ca9 The initial commit of my project

分支切换

要切换到一个已存在的分支,你需要使用git checkout命令。 我们现在切换到新创建的 testing 分支去:这样 HEAD 就指向 testing 分支了。

1
$ git checkout testing

在切换分支之后重新提交,新的提交会提交到新的分支

1
2
$ vim test.rb
$ git commit -a -m 'made a change'

你的testing分支向前移动了,但是master分支却没有,它仍然指向运行git checkout时所指的对象。

现在我们切换回master分支看看:

1
$ git checkout master

这条命令做了两件事。

  • 一是使HEAD指回master分支
  • 二是将工作目录恢复成master分支所指向的快照内容

也就是说,你现在做修改的话,项目将始于一个较旧的版本。本质上来讲,这就是忽略testing分支所做的修改,以便于向另一个方向进行开发。

需要注意的是,分支切换会改变你工作目录中的文件,

在切换分支时,一定要注意你工作目录里的文件会被改变。即便有被跟踪但是没有提交的文件会被自动覆盖掉,如果是切换到一个较旧的分支,你的工作目录会恢复到该分支最后一次提交时的样子。

如果Git不能干净利落地完成这个任务,它将禁止切换分支。

1
2
$ vim test.rb
$ git commit -a -m 'made other changes'

现在,这个项目的提交历史已经产生了分叉,因为刚才你创建了一个新分支,并切换过去进行了一些工作,随后又切换回master分支进行了另外一些工作。上述两次改动针对的是不同分支:

你可以在不同分支间不断地来回切换和工作,并在时机成熟时将它们合并起来。而所有这些工作,你需要的命令只有branch、checkout和commit。

你可以简单地使用git log命令查看分叉历史。运行git log --oneline --decorate --graph --al1,它会输出你的提交历史、各个分支的指向以及项目的分支分叉情况。

由于Git的分支实质上仅是包含所指对象校验和(长度为40的SHA-1值字符串)的文件,所以它的创建和销毁都异常高效。创建一个新分支就相当于往一个文件中写入41个字节(40个字符和1个换行符),如此的简单能不快吗?

创建新分支的同时切换过去

通常我们会在创建一个新分支后立即切换过去,这可以用 git checkout -b <newbranchname> 一条命令搞定。

来看一个Demo

分支的新建与合并

  1. 开发某个网站。
  2. 为实现某个新的用户需求,创建一个分支。
  3. 在这个分支上开展工作。

正在此时,你突然接到一个电话说有个很严重的问题需要紧急修补。 你将按照如下方式来处理:

  1. 切换到你的线上分支(production branch)。
  2. 为这个紧急任务新建一个分支,并在其中修复它。
  3. 在测试通过之后,切换回线上分支,然后合并这个修补分支,最后将改动推送到线上分支。
  4. 切换回你最初工作的分支上,继续工作

新建分支

首先,我们假设你正在你的项目上工作,并且在 master 分支上已经有了一些提交。

现在,你已经决定要解决你的公司使用的问题追踪系统中的#53问题。想要新建一个分支并同时切换到那个分支上,你可以运行一个带有-b参数的git checkout命令:

1
2
$ git checkout -b iss53
Switched to a new branch "iss53"

你继续在#53问题上工作,并且做了一些提交。 在此过程中,iss53 分支在不断的向前推进,因为你已经检出到该分支 (也就是说,你的 HEAD 指针指向了 iss53 分支)

1
2
3
$ vim index.html
$ git commit -a -m 'added a new footer [issue 53]'

现在你接到那个电话,有个紧急问题等待你来解决。有了Git的帮助,你不必把这个紧急问题和iss53的修改混在一起,你也不需要花大力气来还原关于53#问题的修改,然后再添加关于这个紧急问题的修改,最后将这个修改提交到线上分支。你所要做的仅仅是切换回master分支

但是,在你这么做之前,要留意你的工作目录暂存区里那些还没有被提交的修改,它可能会和你即将检出的分支产生冲突从而阻止Git切换到该分支。最好的方法是,在你切换分支之前,保持好一个干净的状态。

有一些方法可以绕过这个问题(即,暂存(stashing)和清理(clean))。现在,我们假设你已经把你的修改全部提交了,这时你可以切换回master分支了:

这里需要注意的是一定要切回master分支之后在新建分支,不要在iss53的分支上新建分支。

1
2
$ git checkout master
Switched to branch 'master'

这个时候,你的工作目录和你在开始#53问题之前一模一样,现在你可以专心修复紧急问题了。请牢记:当你切换分支的时候,Git会重置你的工作目录,使其看起来像回到了你在那个分支上最后一次提交的样子。Git会自动添加、删除、修改文件以确保此时你的工作目录和这个分支最后一次提交时的样子一模一样。

接下来,你要修复这个紧急问题。我们来建立一个hotfix分支,在该分支上工作直到问题解决:

1
2
3
4
5
6
$ git checkout -b hotfix
Switched to a new branch 'hotfix'
$ vim index.html
$ git commit -a -m 'fixed the broken email address'
[hotfix 1fb7853] fixed the broken email address
1 file changed, 2 insertions(+)

你可以运行你的测试,确保你的修改是正确的,然后将hotfix分支合并回你的master分支来部署到线上。

这里需要注意,在分支合并的时候,当前分支是合并后的分支,被合并的分支为git merge xxx 指定的分支。要把B合并到A,那么当前分支应该为A,使用git merge B的命令合并

你可以使用git merge命令来达到上述目的:

1
2
3
4
5
6
$ git checkout master
$ git merge hotfix
Updating f42c576..3a0874c
Fast-forward
index.html | 2 ++
1 file changed, 2 insertions(+)

Fast-forward :在合并的时候,你应该注意到了“快进(fast-forward)”这个词。由于你想要合并的分支hotfix所指向的提交C4是你所在的提交C2的直接后继,因此Git会直接将指针向前移动。

换句话说,当你试图合并两个分支时,如果顺着一个分支走下去能够到达另一个分支,那么Git在合并两者的时候,只会简单的将指针向前推进(指针右移),因为这种情况下的合并操作没有需要解决的分歧——这就叫做“快进(fast-forward)”。

现在,最新的修改已经在master分支所指向的提交快照中,你可以着手发布该修复了。

关于这个紧急问题的解决方案发布之后,你准备回到被打断之前时的工作中。然而,你应该先删除hotfix分支,因为你已经不再需要它了,它master分支已经指向了同一个位置。你可以使用带-d选项的git branch命令来删除分支

1
2
$ git branch -d hotfix
Deleted branch hotfix (3a0874c).

现在你可以切换回你正在工作的分支继续你的工作,也就是针对 #53 问题的那个分支(iss53 分支)。

1
2
3
4
5
6
$ git checkout iss53
Switched to branch "iss53"
$ vim index.html
$ git commit -a -m 'finished the new footer [issue 53]'
[iss53 ad82d7a] finished the new footer [issue 53]
1 file changed, 1 insertion(+)

你在hotfix分支上所做的工作并没有包含到iss53分支中。如果你需要拉取hotfix所做的修改

你可以使用git merge master命令将master分支合并入iss53分支,或者你也可以等到iss53分支完成其使命,再将其合并回master分支

假设你已经修正了#53问题,并且打算将你的工作合并入master分支。为此,你需要合并iss53分支到master分支,这和之前你合并hotfix分支所做的工作差不多。你只需要检出到你想合并入的分支,然后运行git merge命令

1
2
3
4
5
6
$ git checkout master
Switched to branch 'master'
$ git merge iss53
Merge made by the 'recursive' strategy.
index.html | 1 +
1 file changed, 1 insertion(+)

这和你之前合并hotfix分支的时候看起来有一点不一样。在这种情况下,你的开发历史从一个更早的地方开始分叉开来(diverged)。因为,master分支所在提交并不是iss53分支所在提交的直接祖先,Git不得不做一些额外的工作。

出现这种情况的时候,Git会使用两个分支的未端所指的快照(C4和C5)以及这两个分支的公共祖先(C2),做一个简单的三方合并

和之前将分支指针向前推进所不同的是,Git将此次三方合并的结果做了一个新的快照并且自动创建一个新的提交指向它。这个被称作一次合并提交,它的特别之处在于他有不止一个父提交

既然你的修改已经合并进来了,就不再需要iss53分支了。现在你可以在任务追踪系统中关闭此项任务,并删除这个分支。

1
$ git branch -d iss53

上面讲到有一些方法可以绕过这个问题(即,暂存(stashing)和clean),我们来看下

贮藏与清理

有时,当你在项目的一部分上已经工作一段时间后,所有东西都进入了混乱的状态, 而这时你想要切换到另一个分支做一点别的事情。 问题是,你不想仅仅因为过会儿回到这一点而为做了一半的工作创建一次提交。 针对这个问题的答案是git stash命令。

贮藏(stash)会处理工作目录的脏的状态——即跟踪文件的修改与暂存的改动——然后将未完成的修改保存到一个栈上,而你可以在任何时候重新应用这些改动(甚至在不同的分支上)

贮藏工作

运行 git status,可以看到有改动的状态:

1
2
3
4
5
6
7
8
9
$ git status
Changes to be committed:
(use "git reset HEAD <file>..." to unstage)
modified: index.html
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git checkout -- <file>..." to discard changes in working
directory)
modified: lib/simplegit.rb

现在想要切换分支,但是还不想要提交之前的工作;所以贮藏修改。将新的贮藏推送到栈上,运行git stashgit stash push

1
2
3
4
5
6
$ git stash
Saved working directory and index state \
"WIP on master: 049d078 added the index file"
HEAD is now at 049d078 added the index file
(To restore them type "git stash apply")

可以看到工作目录是干净的了:

1
2
3
$ git status
# On branch master
nothing to commit, working directory clean

此时,你可以切换分支并在其他地方工作;你的修改被存储在栈上。要查看贮藏的东西,可以使用git stash list:

1
2
3
4
5
$ git stash list
stash@{0}: WIP on master: 049d078 added the index file
stash@{1}: WIP on master: c264051 Revert "added file_size"
stash@{2}: WIP on master: 21d80a5 added number to log

在本例中,有两个之前的贮藏,所以你接触到了三个不同的贮藏工作。可以通过原来 stash命令的帮助提示中的命令将你刚刚贮藏的工作重新应用:git stash apply

如果想要应用其中一个更旧的贮藏,可以通过名字指定它,像这样:git stash apply stashe@{2}。如果不指定一个贮藏,Git认为指定的是最近的贮藏:

1
2
3
4
5
6
7
8
9
$ git stash apply
On branch master
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git checkout -- <file>..." to discard changes in working
directory)
modified: index.html
modified: lib/simplegit.rb
no changes added to commit (use "git add" and/or "git commit -a")

通过stash存储修改和新建分支比起来,一大优点是,它在应用时并不是必须要有一个干净的工作目录,或者要应用到同一分支才能成功应用贮藏。

stash可以在一个分支上保存一个贮藏,切换到另一个分支,然后尝试重新应用这些修改。当应用贮藏时工作目录中也可以有修改与未提交的文件——如果有任何东西不能干净地应用,Git会产生合并冲突

文件的改动被重新应用了,但是之前暂存的文件(add)却没有重新暂存。想要那样的话,必须使用--index选项来运行git stash apply命令,来尝试重新应用暂存的修改。如果已经那样做了,那么你将回到原来的位置:

1
2
3
4
5
6
7
8
9
10
$ git stash apply --index
On branch master
Changes to be committed:
(use "git reset HEAD <file>..." to unstage)
modified: index.html
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git checkout -- <file>..." to discard changes in working
directory)
modified: lib/simplegit.rb

应用选项只会尝试应用贮藏的工作——在堆栈上还有它。可以运行git stash drop加上将要移除的贮藏的名字来移除它:

1
2
3
4
5
6
$ git stash list
stash@{0}: WIP on master: 049d078 added the index file
stash@{1}: WIP on master: c264051 Revert "added file_size"
stash@{2}: WIP on master: 21d80a5 added number to log
$ git stash drop stash@{0}
Dropped stash@{0} (364e91f3f268f0900bc3ee613f9f733e82aaed43)

可能有小伙伴会说,如过stash之后,我忘记了有过stash的操作,通过status命令想看到文件状态?

1
2
3
$ git status -s
M index.html
M lib/simplegit.rb

git stash命令的--keep-index选项。它告诉Git不仅要贮藏所有已暂存的内容,同时还要将它们保留在索引中。,即我们可以通过status命令开查看状态

1
2
3
4
$ git stash --keep-index
Saved working directory and index state WIP on master: 1b65b17 added the
index file
HEAD is now at 1b65b17 added the index file
1
2
$ git status -s
M index.html

默认情况下,git stash只会贮藏已修改暂存的已跟踪文件。如果指定--include-untracked-u选项,Git也会贮藏任何未跟踪文件

1
2
3
4
5
6
7
8
9
10
$ git status -s
M index.html
M lib/simplegit.rb
?? new-file.txt
$ git stash -u
Saved working directory and index state WIP on master: 1b65b17 added the
index file
HEAD is now at 1b65b17 added the index file
$ git status -s
$

然而,在贮藏中包含未跟踪的文件仍然不会包含明确忽略的文件。要额外包含忽略的文件,请使用--al1-a选项。

从贮藏创建一个分支

如果贮藏了一些工作,将它留在那儿了一会儿,然后继续在贮藏的分支上工作,在重新应用工作时可能会有问题。如果应用尝试修改刚刚修改的文件,你会得到一个合并冲突并不得不解决它。

如果想要一个轻松的方式来再次测试贮藏的改动,可以运行git stash branch <new branchname>以你指定的分支名创建一个新分支,检出贮藏工作时所在的提交,重新在那应用工作,然后在应用成功后丢弃贮藏:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
$ git stash branch testchanges
M index.html
M lib/simplegit.rb
Switched to a new branch 'testchanges'
On branch testchanges
Changes to be committed:
(use "git reset HEAD <file>..." to unstage)
modified: index.html
Changes not staged for commit:
(use "git add <file>..." to update what will be committed)
(use "git checkout -- <file>..." to discard changes in working
directory)
modified: lib/simplegit.rb
Dropped refs/stash@{0} (29d385a81d163dfd45a452a2ce816487a6b8b014)

这是在新分支轻松恢复贮藏工作并继续工作的一个很不错的途径

清理工作目录

对于工作目录中一些工作或文件,你想做的也许不是贮藏而是移除。git clean命令就是用来干这个的。

需要谨慎地使用这个命令,因为它被设计为从工作目录中移除未被追踪的文件。

如果你改变主意了,你也不一定能找回来那些文件的内容。 一个更安全的选项是运行 git stash --all 来移除每一样东西并存放在栈中。

你可以使用git clean命令去除冗余文件或者清理工作目录。 使用git clean -f -d命令来移除工作目录中所有未追踪的文件以及空的子目录。 -f 意味着“强制(force)”或“确定要移除”,使用它需要 Git 配置变量 clean.requireForce 没有显式设置为 false。

如果只是想要看看它会做什么,可以使用 --dry-run -n 选项来运行命令, 这意味着“做一次演习然后告诉你 将要 移除什么”。

1
2
3
$ git clean -d -n
Would remove test.o
Would remove tmp/

默认情况下,git clean 命令只会移除没有忽略的未跟踪文件。

1
2
3
4
$ git status -s
M lib/simplegit.rb
?? build.TMP
?? tmp/

任何与 .gitignore 或其他忽略文件中的模式匹配的文件都不会被移除。 如果你也想要移除那些文件,

1
2
3
$ git clean -n -d
Would remove build.TMP
Would remove tmp/

例如为了做一次完全干净的构建而移除所有由构建生成的 .o 文件, 可以给 clean 命令增加一个-x选项。

1
2
3
4
$ git clean -n -d -x
Would remove build.TMP
Would remove test.o
Would remove tmp/

分支管理

git branch 命令不只是可以创建与删除分支。 如果不加任何参数运行它,会得到当前所有分支的一个列表:

git branch -a 命令可以查看 本地和远程的分支

注意 master 分支前的 * 字符:它代表现在检出的那一个分支(也就是说,当前 HEAD 指针所指向的分支)。

这意味着如果在这时候提交,master 分支将会随着新的工作向前移动。 如果需要查看每一个分支的最后一次提交,可以运行 git branch -v 命令:

--merged与--no-merged这两个有用的选项可以过滤这个列表中已经合并尚未合并当前分支的分支

  • 查看哪些分支已经合并到当前分支,可以运行git branch--merged:可以使用git branch-d 删除掉;你已经将它们的工作整合到了另一个分支,所以并不会失去任何东西。
  • 查看所有包含未合并工作的分支,可以运行 git branch --no-merged:因为它包含了还未合并的工作,尝试使用git branch-d命令删除它时会失败:如果真的想要删除分支并丢掉那些工作,如同帮助信息里所指出的,可以使用-D选项强制删除它。

嗯,时间关系,先整理到这里,关于分支工作流,变基等在之后的博文中和小伙伴们分享

博文整理参考


《Pro Git》

发布于

2022-07-26

更新于

2023-06-21

许可协议

评论
Your browser is out-of-date!

Update your browser to view this website correctly.&npsb;Update my browser now

×