如果大家平常都在使用 Git 作为版本控制工具的话,那么一定每天都能见到 origin
,诸如:
$ git push origin main
$ git fetch origin main
$ git pull origin main
# ...
这里的 origin
,还有看似相同的 origin/master
、origin/main
又是什么呢?
一、远程名称(Remote Name)
在 Git 中,其实无论是 origin
,还是 upstream
并没有特殊的含义,但由于被广泛使用,因此它们有了约定俗成、众所周知的含义。
就好比如说,在现实世界中小明、小红是再普通不过的名字,但由于在小学语文课本的对话中常被作为男一女一,用于表示对话的两个人而已,并没有特别的意义在里面。而在技术博文中,经常可以看到使用 foo
、bar
作为变量标识符举例,它们就相当于语文课本中的小明、小红一样。那么本文接下来要讨论的 origin
、upstream
等也是同样的道理。
先来个餐前菜...
如果我跟你说,以下两条命令是完全等效的,你是不是就差不多猜得出 origin
表示什么了?
$ git push origin main
$ git push git@github.com:toFrankie/repo-demo.git main
是的,跟你猜的一样...
1.1 Origin
我们用示例来讲...
先在本地随意创建一个 Git 仓库 repo-demo
,然后新增一个 README.md
文件,接着 Commit 一下(如下图):
以上都没问题!接着,我们试着 Push 一下:
可以看到 git push
失败了,原因很容易理解:我们只是在本地创建一个仓库,并没有将本仓库与远程仓库进行关联,因此 Git 无法理解是将其推送至哪个代码托管平台,然后也不知道是平台上的哪个远程仓库,是 GitHub 平台的,还是 GitLab 平台的?是平台上的 React 仓库,还是 Vue 仓库,还是别的什么仓库?Git 统统都不知道,那么自然是无法替你办事了。
因此,我们需要做的就是把本地的 repo-demo
仓库与远程仓库关联一下(请注意,一个本地仓库是可以关联多个远程仓库的):
$ git remote add origin <repo_address>
这里用到了 origin
,我们先不管为什么用 origin
,用其他(比如 foo
)行不行的问题?(答案是可以的)
关联之后,再进行 Push 就能成功了。
那么 git remote add
内部做了什么默默无闻的工作呢,它其实是往 .git/config
中写入了一个叫 [remote "origin"]
配置:
[core]
repositoryformatversion = 0
filemode = true
bare = false
logallrefupdates = true
ignorecase = true
precomposeunicode = true
[remote "origin"]
url = git@github.com:toFrankie/repo-demo.git
fetch = +refs/heads/*:refs/remotes/origin/*
如果你本地的仓库是通过 git clone
下来的,Git 会默认将远程仓库命名为 origin
,自动帮你关联上远端仓库(可在 .git/config
文件中看到已有 [remote "origin"]
配置项了),因此 Commit 之后就能直接 Push 了。
When a repo is cloned, it has a default remote called
origin
that points to your fork on GitHub, not the original repo it was forked from.(引自 Github page)
如果我们在 GitHub 新创建一个 Repository 的话,会看到以下指引:
我们来分析一下,这配置表示什么意思。
[remote "origin"]
url = git@github.com:toFrankie/repo-demo.git
fetch = +refs/heads/*:refs/remotes/origin/*
通过 git remote add
命令,添加了一个叫做 origin
的远程名称(Remote Name),
- 其中
url
参数,表示该远程名称对应的远程仓库地址。 - 其中
fetch
参数分为两部分,以冒号:
进行分割,冒号左边表示本地仓库文件夹,冒号右边表示远程仓库在本地的副本文件夹。里面的加号+
表示往里面添加数据的意思。
当使用 git fetch origin
时,Git 将远程仓库下的所有分支拉取到本地的 refs/remotes/origin/
目录下,然后 git merge
时,它会把 refs/remotes/origin/
目录下的对应分支合并到 refs/heads/
目录下对应分支上。
那么 origin
究竟是什么呢?
请注意,
origin
只是一个名称(别名),用于指向远程仓库。这个别名是可以自行修改的,比如命名为foo
、bar
等。使用别名好处是「方便」。
比起记住一个远程仓库地址,别名实在方便太多了。将 origin
作为远程仓库的别名是较为普遍的做法,况且所有代码托管平台默认就是 origin
。
回到文章开头的例子:
$ git push origin main
# 相当于(其中 origin 指向了 git@github.com:toFrankie/repo-demo.git 远程仓库)
$ git push git@github.com:toFrankie/repo-demo.git main
以上两种方式是完全等价的,这样就更能体现别名的优势了,简洁很多。
既然是别名,自然是可以修改的,主要有以下命令:
# 新增远程名称(一个本地仓库,可以关联多个远程仓库)
$ git remote add <remote-name> <repo-address>
# 删除已存在远程名称(只会移除本地仓库与远程仓库的管理,不会删除远程仓库的代码哈)
$ git remote rm <remote-name>
# 更新远程名称关联的远程仓库
$ git remote set-url <remote-name> <repo-address>
# 修改远程名称(也可以先删除再添加)
$ git remote rename <old-remote-name> <new-remote-name>
比如,像这样:
然后,我们修改下远程名称为 foo
,也是可以的:
接着,我们随意修改个文件 Push 一下,是这样的 git push foo main
:
到这里,你应彻底明白 origin
是什么了吧。
前面提到过,一个本地仓库是可以关联多个远程仓库的,举个例子:
$ git remote add bar git@github.com:toFrankie/git-test-demo.git
我们可以查看下 .git/congfig
配置文件,如下(或者通过 git remote -v
查看)
从图中可以看到,别名 foo
和 bar
分别指向了两个不同的远程仓库,然后使用方法与 origin
是相同的,比如:
# 将本地的 main 分支推送至 foo 对应的远程仓库(repo-demo)
$ git push foo main
# 将本地的 main 分支推送至 bar 对应的远程仓库(git-test-demo)
$ git push bar main
1.2 upstream(一个特殊的 remote name)
upstream
的译为“上游”。当你 git clone
一个别人的 Repository 到本地,由于你不是该仓库的成员,因此你是无法向该仓库推送代码的。此时,相较于本地仓库,别人的这个 Repository 称为 upstream
。
我们可以 Fork 这个 Repository 到自己 GitHub 账号下,然后通过 git clone
将这个 Fork 出来的仓库克隆到本地电脑上。(下文将这个别人的仓库称为 Upstream-Repo
,Fork 出来的仓库称为 Origin-Repo
)
大致关系如图所示(源自),其中 Upstream-Repo
对应图中的 Original,Origin-Repo
对应图中 Fork:
当我们将 Origin-Repo
克隆到本地,Git 会默认创建一个 origin
的别名指向 Origin-Repo
的仓库地址。
如果要跟踪 Upstream-Repo
仓库的变更,您需要添加另一个名为 upstream
的别名,使其指向 Upstream-Repo
仓库。
# 1. 添加上游仓库的别名
$ git remote add upstream <upstream-repo-address>
# 2. 获取上游仓库的变更
$ git fetch upstream
# 3. 有需要的话,可以通过 merge 或 rebase 方式合并到本地分支中,比如:
$ git merge upstream/main
尽管添加了 upstream
,诸如 git push upstream main
等方式试图向 Upstream-Repo
提交代码仍然是不被允许的,因为你不是 Upstream-Repo
仓库的成员。想给 Upstream-Repo
仓库贡献代码的话,只能通过 Pull Request 的方式。
当然,这一节提到的 upstream
也是一个约定俗称的别名,也是可以自定义的。
1.3 小结
除了 origin
、upstream
等有众所周知的含义的远程名称之外,我们还可以这样使用:
由于一个本地仓库是可以关联多个远程仓库的,因此,可以设置多个「别名」分别指向不同的远程仓库(比如一个 GitHub、一个 GitLab、一个 Gitee),然后通过别名的方式方便、快速地拉取某个远程仓库的代码或者将代码推送至某个远程仓库。
# 添加 github 别名
$ git remote add github git@github.com:toFrankie/repo-demo.git
# 添加 gitlab 别名
$ git remote add gitlab git@gitlab.com:toFrankie/repo-demo.git
# 添加 gitee 别名
$ git remote add gitee git@gitee.com:toFrankie/repo-demo.git
小结一下:
常见的
origin
、upstream
都只是通过git remote add
命令创建的名称(Remote Name),用于指向某个远程仓库(Remote URL)。常用
origin
作为远程仓库的别名,是一个较为主流的做法。同时,也是各大代码托管平台的默认名称(即git clone
一个远程仓库, Git 会默认将origin
指向该仓库)。如果你觉得不爽,完全可以自定义(git remote set-url
)为“阿猫”、“阿狗”等名称。查看本地仓库关联的远程仓库信息,可以在
.git/config
文件或通过git remote -v
命令查看。使用别名的最大好处是,无需记住远程仓库的 URL,也是唯一的好处吧。不用也是完全 OK 的,完全可以直接使用远程仓库 URL,但我想不会有这种朋友吧。
(建议)若无特殊需求,不要为了个性,试图更改 origin
、upstream
等被广泛使用的别名,其中所表示的约定俗成的、众所周知的含义。
二、远程分支(Remote Branch)
常说的「远程分支」是远程仓库对应分支在本地的一个副本。比如常见的 origin/master
、origin/main
、origin/develop
等都是远程分支,可以在 .git/refs/remotes/
目录下看到。
# 查看所有本地分支
$ git branch
# 查看所有远程分支(-r 是 --remotes 的简写)
$ git branch -r
# 查看所有本地分支和远程分支(-a 是 --all 的简写)
$ git branch -a
可以通过 git branch -r
命令查看所有的远程分支:
frankie@iMac repo-demo % git branch -r
origin/HEAD -> origin/main
origin/dev
origin/main
如果对 origin/HEAD
不理解的话,先不管下文会介绍。
上一节,我们介绍了远程名称只是一个代号、别名,是可以修改的。那么我们将 Remote Name 由 origin
修改为 foo
,那么远程分支,会不会由 origin/main
变为 foo/main
呢?
修改前:
frankie@iMac repo-demo % git remote -v
origin git@github.com:toFrankie/repo-demo.git (fetch)
origin git@github.com:toFrankie/repo-demo.git (push)
frankie@iMac repo-demo % git branch -r
origin/HEAD -> origin/main
origin/dev
origin/main
修改后:
frankie@iMac repo-demo % git remote rename origin foo
frankie@iMac repo-demo % git remote -v
foo git@github.com:toFrankie/repo-demo.git (fetch)
foo git@github.com:toFrankie/repo-demo.git (push)
frankie@iMac repo-demo % git branch -r
foo/HEAD -> foo/main
foo/dev
foo/main
果然,将远程名称修改之后,远程分支名称也会跟着改变的。我们通过 tree
命令看下目录结构,如下:
frankie@iMac repo-demo % tree .git/refs
.git/refs
├── heads
│ ├── dev
│ └── main
├── remotes
│ └── foo
│ ├── HEAD
│ ├── dev
│ └── main
└── tags
那么接下来,若无特殊说明,都将以 origin
作为远程名称进行说明或举例。
通常,拉取最新代码的过程是这样的:
通过
git fetch
拉取代码的过程:先读取.git/config
文件里面的配置[remote <remote-name>]
,将里面的所有(因为fetch
并没有指定其中一个或多个远程仓库)远程名称对应仓库的分支下载到本地,并放在.git/refs/remotes/<remote-name>/
目录下。比如
git fetch origin main
会创建或更新.git/refs/remotes/origin/main
的文件,此时通过git branch -r
就能看到一个origin/main
的分支。但注意,我们使用的时候还是用origin/main
而不是remotes/origin/main
哦。有时候,我们可能会通过
git diff
命令来对比本地分支与远程分支的一些信息,才决定要不要合并。比如,git diff main origin/main
。通过
git merge
或git rebase
来进行分支合并。比如git merge origin/main
,表示将远程分支origin/main
合并至本地分支main
中。
也可以直接使用 git pull
命令,其实包括了 git fetch
和 git merge
两个过程。请注意 git fetch
并不会修改「本地分支」的代码。
细心的同学可能会发现,refs/remotes/origin/
目录下,相应的分支文件记录的只是一个 Commit-ID
(SHA-1),比较特殊的是 HEAD
文件(即 origin/HEAD
分支)记录的是 ref: refs/remotes/origin/main
的东西,它始终指向默认远程分支。
三、HEAD、Detached HEAD、origin/HEAD、FETCH_HEAD、ORIG_HEAD 区别
这个对于刚接触的同学,可能看起来有点懵。
其实 Git 中的「分支」是由一个或多个 Commit-ID
组成的集合。通过 git branch
命令创建的分支,只是对某个 Commit-ID
的「引用」。因此,使用 git branch -d
删除某个本地分支,也只是删除了这个「引用」而已,并不会删除任何的 Commit-ID
。但是,如果一个 Commit-ID
没有被任何一个分支引用的话,在一定时间之后,将会被 Git 回收机制删除。
本节内容将会讲述以下相关内容:
HEAD
跟「本地分支」相关。Detached HEAD
是一种特殊状态的HEAD
。origin/DEAD
跟「远程分支」相关。FETCH_HEAD
跟git fetch
操作相关ORIG_HEAD
跟git merge
、git reset
等「危险操作」相关
3.1 HEAD
我们在哪能看到 HEAD
呢?
接着,我们从 main
分支切换至 dev
分支。
对比发现,HEAD
发生改变了。
前面提到,分支只是对 Commit-ID 的引用。每当在某个分支上提交代码,Git 都会产生一个全新的、唯一的 Commit-ID,此时我们的分支名称也随之移向最新的一个 Commit-ID。
关于 HEAD 存放于本地仓库的 .git/HEAD
文件里面,利用 cat
命令可以看到它的内容。
frankie@iMac repo-demo % cat .git/HEAD
ref: refs/heads/dev
frankie@iMac repo-demo % cat .git/refs/heads/dev
866bc9f1d8f4797c0e46e959cb0c9abdd47d8176
所以说到底,此时 HEAD
只是对 Commit-ID 为 866bc9f1d8f4797c0e46e959cb0c9abdd47d8176
的引用。如果切回 main
分支,那么 HEAD
相应的内容就会跟着改变。
而 HEAD
则是比较特殊的一个引用(有些文章称为「指针」,也问题不大)。除了 git commit
之外,git checkout
、git reset
等命令都会影响 HEAD
的指向。
一句话总结:HEAD 是对当前 Commit-ID 的「引用」。
- 当使用
git commit
时,HEAD
会跟着移动,并指向最新的 Commit-ID。- 当使用
git checkout
时,HEAD
会移动并指向对应分支的最新一个 Commit-ID。- 当使用
git reset
时,HEAD
会移动至对应分支的某个 Commit-ID。请注意git reset --hard
可以将HEAD
和Branch
移动至任何地方。
顺道提一下,git reset
的本质就是移动 HEAD
来达到撤销的目的。
观察以下示例,我使用 git reset --soft
将 HEAD
从 Commit-ID 为 42d46a2
移至 222a33c
,变化如下:
frankie@iMac repo-demo % git log
commit 42d46a2356cfdde0ad80bfc042b6cb15eae04759 (HEAD -> main, origin/main, origin/HEAD)
Author: Frankie <1426203851@qq.com>
Date: Sat Feb 26 16:44:30 2022 +0800
docs: update
commit e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:55:48 2022 +0800
docs: update readme.md
commit 222a33cb3185457a5d726325aa7233e43f0b92d3
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:36:51 2022 +0800
docs: update README.md
commit 62733124c6485bc5d81123dd3eae95d4da22753f
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 15:45:15 2022 +0800
docs: add README.md
frankie@iMac repo-demo % git reset --soft 222a33cb3185457a5d726325aa7233e43f0b92d3
frankie@iMac repo-demo % git log
commit 222a33cb3185457a5d726325aa7233e43f0b92d3 (HEAD -> main)
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:36:51 2022 +0800
docs: update README.md
commit 62733124c6485bc5d81123dd3eae95d4da22753f
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 15:45:15 2022 +0800
docs: add README.md
此时,我们再看下 .git/HEAD
的内容:
frankie@iMac repo-demo % cat .git/HEAD
ref: refs/heads/main
frankie@iMac repo-demo % cat .git/refs/heads/main
222a33cb3185457a5d726325aa7233e43f0b92d3
关于
git reset
的三个参数--mixed
(默认)、--hard
、--soft
的区别,推荐看这篇文章,讲得很详细易懂。
3.2 Detached HEAD
detached HEAD
可以称为「游离 HEAD」,也有称为「分离 HEAD」的。
一般情况下,我们的 HEAD
会指向某个分支的某个 Commit-ID。但是 HEAD
偶尔会发生「没有指向某个本地分支」的情况,这种状态的 HEAD
称为 detached HEAD
。
以下情况,就可能会出现 detached HEAD
:
- 使用
git checkout
跳转至某个 Commit-ID,而这个 Commit-ID 刚好目前没有分支指向它。- Rebase 的过程其实也是处于不断的 detached HEAD 状态。
- 切换至某个远程分支的时候。
我们先将 Git 的提示语言切换为英文,看得更加清晰。
frankie@iMac repo-demo % git checkout e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
Note: switching to 'e0c619ca3978a38f6eabe79c3dfc67d4296ccc36'.
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 switching back to a branch.
If you want to create a new branch to retain commits you create, you may
do so (now or later) by using -c with the switch command. Example:
git switch -c <new-branch-name>
Or undo this operation with:
git switch -
Turn off this advice by setting config variable advice.detachedHead to false
HEAD is now at e0c619c docs: update readme.md
此时 HEAD
指向 e0c619c
,这个就是 detached HEAD
。
还有,前面提到「没有指向某个本地分支」,但其实我们使用 git branch
会发现有以下这样一个分支:
frankie@iMac repo-demo % git branch
* (HEAD detached at e0c619c)
dev
main
但注意,当我们切换至其他分支时,这个 (HEAD detached at e0c619c)
分支是会被干掉的,因为它只是临时的。因此人家也提醒你可以使用 git switch -c <new-branch-name>
命令,以创建一个新分支来指向该 Commit-ID。
假设我们有这样一个场景:想要查看某个历史版本的源码,就可以利用此功能来解决。
# -t 即 --track
$ git branch <new-branch-name> <commit-id>
# 或者
$ git checkout -b <new-branch-name> <commit-id>
如果我们使用 git checkout origin/main
切换至 origin/main
远程分支时,也会产生一个 detached HEAD
的。如果我们想基于某个远程分支,新建一个同名本地分支,可以这样:
$ git checkout -t origin/dev
branch 'dev' set up to track 'origin/dev'.
Switched to a new branch 'dev'
# 相当于
$ git checkout -b dev origin/dev
如果要离开 detached HEAD
状态很简单,只要切换至其他分支即可。
3.3 origin/HEAD
从名字可以看出,origin/HEAD
也是一个「远程分支」,其中 origin
则对应远程名称。
一般情况下,origin/HEAD
总是指向远程仓库的「默认分支」。假设我们的远程默认分支为 main
。那么在远程仓库在本地的副本,origin/HEAD
就是相当于 origin/main
。
Git 提供了以下命令让我们去修改 origin/HEAD
的指向(可通过 git remote set-head -h
查看):
# 将 origin/HEAD 设为远程仓库的默认分支(-a 即 --auto)
$ git remote set-head <remote-name> -a
# 将 origin/HEAD 设为某个远程分支,
# 比如 git remote set-head origin dev,将 origin/HEAD 指向远程的 dev 分支,相当于 origin/dev
$ git remote set-head <remote-name> <branch-name>
# 删除 origin/HEAD(-d 即 --delete)
$ git remote set-head <remote-name> -d
以上注释部分,假定了远程名称为 origin
。
我们修改下 origin/HEAD
为远程仓库的 dev
分支,origin/HEAD
文件里面存储的内容,同样表示的也是对某个远程分支的引用,仅此而已。
frankie@iMac repo-demo % git remote set-head origin dev
frankie@iMac repo-demo % cat .git/refs/remotes/origin/HEAD
ref: refs/remotes/origin/dev
3.4 ORIG_HEAD(拓展内容)
在 .git
目录下,有一个 ORIG_HEAD
的文件,你有没有好奇怪,它是什么呢?
frankie@iMac repo-demo % cat .git/ORIG_HEAD
222a33cb3185457a5d726325aa7233e43f0b92d3
还记得,前面使用过 git reset --soft 222a33cb3185457a5d726325aa7233e43f0b92d3
指令来移动 HEAD
的指向吗?
当我们进行了一些「危险操作」时,比如 git reset
、git merge
、git rebase
等操作时,Git 会将当前 HEAD
指向的的 Commit-ID 原值保存至 ORIG_HEAD
文件内。需要注意的是,类似 git commit
等操作并不会更新 ORIG_HEAD
的内容。
这样的话,加入我们执行了一些「误操作」时,可以利用 git reset --hard ORIG_HEAD
回退至上一步。
举个例子,我们进行一次 git reset
操作:
frankie@iMac repo-demo % git log
commit 42d46a2356cfdde0ad80bfc042b6cb15eae04759 (HEAD -> main, origin/main)
Author: Frankie <1426203851@qq.com>
Date: Sat Feb 26 16:44:30 2022 +0800
docs: update
commit e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:55:48 2022 +0800
docs: update readme.md
commit 222a33cb3185457a5d726325aa7233e43f0b92d3
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:36:51 2022 +0800
docs: update README.md
commit 62733124c6485bc5d81123dd3eae95d4da22753f
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 15:45:15 2022 +0800
docs: add README.md
在 reset
之前,我们可以看到 HEAD
指向的 Commit-ID 为 42d46a2356cfdde0ad80bfc042b6cb15eae04759
,接着我们执行 git reset 指令:
frankie@iMac repo-demo % git reset --soft e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
frankie@iMac repo-demo % git log [16:39:25]
commit e0c619ca3978a38f6eabe79c3dfc67d4296ccc36 (HEAD -> main)
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:55:48 2022 +0800
docs: update readme.md
commit 222a33cb3185457a5d726325aa7233e43f0b92d3
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:36:51 2022 +0800
docs: update README.md
commit 62733124c6485bc5d81123dd3eae95d4da22753f
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 15:45:15 2022 +0800
docs: add README.md
在 git reset
完成之后,HEAD
指向了 e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
。
既然我们前面提到过,ORIG_HEAD
会记录高危操作前的 Commit-ID。如果没错的话,此时 ORIG_HEAD
记录的应该是 42d46a2356cfdde0ad80bfc042b6cb15eae04759
。
frankie@iMac repo-demo % cat .git/ORIG_HEAD
42d46a2356cfdde0ad80bfc042b6cb15eae04759
如果我们想吃后悔药了,就可以通过 git reset --hard ORIG_HEAD
进行回退:
frankie@iMac repo-demo % git reset --hard ORIG_HEAD
HEAD is now at 42d46a2 docs: update
frankie@iMac repo-demo % git log
commit 42d46a2356cfdde0ad80bfc042b6cb15eae04759 (HEAD -> main, origin/main)
Author: Frankie <1426203851@qq.com>
Date: Sat Feb 26 16:44:30 2022 +0800
docs: update
commit e0c619ca3978a38f6eabe79c3dfc67d4296ccc36
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:55:48 2022 +0800
docs: update readme.md
commit 222a33cb3185457a5d726325aa7233e43f0b92d3
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 18:36:51 2022 +0800
docs: update README.md
commit 62733124c6485bc5d81123dd3eae95d4da22753f
Author: Frankie <1426203851@qq.com>
Date: Sun Feb 20 15:45:15 2022 +0800
docs: add README.md
当然,不同的场景下版本回退还有其他方式的,视乎实际场景,这里就不展开了。
3.5 FETCH_HEAD
其中 FETCH_HEAD
与 git fetch
有关,也是关键部分。
FETCH_HEAD
指的是某个分支在远程仓库上最新的状态。每一个执行过git fetch
操作的本地仓库都会存在一个FETCH_HEAD
列表,这个列表保存在.git/FETCH_HEAD
文件中。FETCH_HEAD
文件中的每一行对应着远程仓库的一个分支。当前本地分支指向的FETCH_HEAD
就是该文件中的「第一行」对应的分支(这段表述源于此处)。
我们知道 git fetch
用以下几种用法:
# 1
$ git fetch
# 2
$ git fetch <remote-name>
# 3
$ git fetch <remote-name> <remote-branch-name>
# 4
$ git fetch <remote-name> <remote-branch-name>:<local-branch-name>
- git fetch
拉取「所有远程仓库」所包含的分支到本地,并在本地创建或更新远程分支。所有分支最新的 Commit-ID 都会记录在 .git/FETCH_HEAD
文件中,若有多个分支,FETCH_HEAD
内会多行数据。
- git fetch origin
拉取 origin
对应的远程仓库的所包含的分支到本地,FETCH_HEAD
设定同上。
- git fetch origin main
拉取 origin
对应远程仓库的 main
分支到本地,且 FETCH_HEAD
只记录了一条数据,那就是远程仓库 main 分支最新的 Commit-ID。
- git fetch origin main:temp
拉取 origin
对应远程仓库的 main
分支到本地,其中 FETCH_HEAD
记录了远程仓库 main
分支最新的 Commit-ID,并且基于远程仓库的 main
分支创建一个名为 temp
的新本地分支(但不会切换至新分支)。
因此,
FETCH_HEAD
记录的是从远程仓库拉取到本地,「对应分支」的最新一个 Commit-ID。当通过git fetch
拉取代码时,
- 若有具体指定了某个远程仓库的某个分支,那么
FETCH_HEAD
就对应此分支。- 若没有具体指定远程仓库的某个分支,
a.FETCH_HEAD
总是指向.git/FETCH_DEAD
首行对应的分支。
b. 文件.git/FETCH_DEAD
可能会记录着多个分支,且该文件首行对应的是git fetch
时所在分支的同名远程分支。
接着,与 FETCH_HEAD
相关的是 git pull
操作。
git pull
等价于 git fetch
+ git merge FETCH_HEAD
两个步骤的结合。
当 git pull
不添加其他参数时,等价于 git pull <remote-name> <当前分支名>
,如果远程仓库无与之对应的同名分支,执行该命令就会抛出错误。举个例子:
frankie@iMac repo-demo % git branch -a
* temp
remotes/origin/HEAD -> origin/dev
remotes/origin/dev
remotes/origin/main
frankie@iMac repo-demo % git pull
There is no tracking information for the current branch.
Please specify which branch you want to merge with.
See git-pull(1) for details.
git pull <remote> <branch>
If you wish to set tracking information for this branch you can do so with:
git branch --set-upstream-to=origin/<branch> temp
好,我们切换到本地的 main
分支,远程仓库有与之对应的同名分支。
# 相当于 git pull origin main
$ git pull
拆分为以下步骤:
git fetch origin main
将远程仓库的main
分支最新 Commit-ID 记录到.git/FETCH_HEAD
中,此时FETCH_HEAD
指向该 Commit-ID。git merge FETCH_HEAD
将FETCH_HEAD
对应的 Commit-ID 合并至本地main
分支中。如果合并过程不存在冲突(即只是 Fast-Forward),那么可以顺利完成git pull
最后一个步骤,否则的话,需要手动解决冲突。
四、More...
其实上面介绍了很多,细心的同学可能会发现,其实无论是「本地分支」,还是「远程分支」,它们记录的只是一个 Commit-ID 或者是对某个分支的引用(形如 ref: refs/heads/main
)。
我们观察本地仓库的 .git
目录可以发现,我们的本地分支、远程分支、标签都是存在于 .git/refs/
目录下:
frankie@iMac repo-demo % tree .git/refs
.git/refs
├── heads
│ ├── dev
│ └── main
├── remotes
│ └── origin
│ ├── HEAD
│ ├── dev
│ └── main
└── tags
前面介绍过,「分支」是由一个或多个 Commit-ID
组成的集合。但我们合并的确是实实在在的代码啊,那么这些代码被存放到哪呢?
具体数据都被放在 .git/objects/
目录下。
然后,现在回头再看 .git/config
的配置,看起来是不是很容易理解了。
[core]
repositoryformatversion = 0
filemode = true
bare = false
logallrefupdates = true
ignorecase = true
precomposeunicode = true
[remote "origin"]
url = git@github.com:toFrankie/repo-demo.git
fetch = +refs/heads/*:refs/remotes/origin/*
[branch "main"]
remote = origin
merge = refs/heads/main
[branch "dev"]
remote = origin
merge = refs/heads/dev
未完待续...