跳至主要內容

merge origin master 和 merge origin/master的不同之处

AndyBin原创大约 4 分钟

说明

在网上找到的普遍解释如下:

git merge origin master
# 将origin merge 到 master 上

git merge origin/master
# 将origin上的master分支 merge 到当前 branch 上

但是这个解释不合理,origin 默认指向的就是 master 分支,所以 merge origin master 和 merge origin/master 是一样的。

于是我自己创建了一个 git 仓库,来模拟这种场景。结果发现两种用法,似乎效果一样。然后各种查阅资料,同时也在国外的一篇帖子上看到了这种说法:

相关信息

  • All parameters to ‘git merge’ in this case are branches that you’re merging from, i.e. source branches. You’re always merging to the current branch.
  • Because origin is a remote name, git automatically expands it to that remote’s default branch, so it’s actually equivalent to origin/master – the command is being told to merge the same branch twice.
  • (It is possible to merge more than one branch, known as “octopus merge”, but this is rarely done – and when it is done, the branches of course are different.)
  • As it is, the command doesn’t make much sense. Maybe it should have been either git merge origin/master (without the duplication) or git pull origin master.

这个说法我比较认可。但是怎么验证呢?假设我当前分支为 A,于是我又新建了一个分支 B,通过手动修改,使得远程 master 和本地 B 分支的代码均与 A 分支有差异,现在想把远程 master 分支代码和本地 B 分支代码合并到 A 分支:

git fetch origin master
git merge origin branchB

然后查看代码,发现远程 master 分支的变更和本地 B 分支的变更都合并到 A 分支上了。这里的 origin 指的是远程库,因为没写具体名字,所以指向默认分支 master,实际上等同于:

git merge origin/master branchB

查看 Git 帮助文档:git merge --help 就可以发现:

git-merge - Join two or more development histories together

git merge 可以合并多个分支,即该命令后面可以跟多个分支的名字,都是将这些分支的变更合并到当前分支。

所以网上说的 git merge origin master 是把 origin merge 到 master 上的说法是错误的。

实际是把远程分支 master 在本地的副本以及本地分支 master 合并到当前分支 A。

其实直接使用 git pull origin master 更简洁,相当于 git fetch origin master 和 git merge origin/master。

这个其实不难理解,git merge origin master 其实是远程 origin 默认的分支和本地的 master 分支合并到当前分支上,合并了两个分支;而 git merge origin/master 其实是本地的远程 origin 仓库 master 分支的拷贝合并到当前分支上,合并了一个分支。

补充

origin 的由来

当你执行 git clone 的时候,就会产生一个 origin,例如:

git clone https://github.com/git/git.git

然后我们执行git remote,就可以看到一个 origin:

$ git remote
origin

这个 origin 代表什么呢?它其实代表的是远程 git 服务器的一个简称,我们执行git remote -v:

$ git remote -v
origin git@github.com:git/git.git (fetch)
origin git@github.com:git/git.git (push)

也就是说这里 origin 代表的是一个远程 git 服务器的地址,俗称远程主机名。

我么可以通过 git remote rename 去修改它:

$ git remote rename origin testname
$ git remote -v
testname git@github.com:git/git.git (fetch)
testname git@github.com:git/git.git (push)

现在 testname 就代表了远程服务器地址。

origin/master 的由来

当我们执行 git fetch 命令的时候:

$ git fetch <远程主机名> <分支名>
git fetch origin master

会从 git 服务器上(origin 代表远程 git 服务器地址),拉取了 master 分支到本地,本地就会新建一个“远程主机名/分支名”,就是 origin/master。

如果你修改了远程主机名为 testname,那么对应的本地分支就是 testname/master。

这个分支不同于你本地开发的 master 分支,它是用来和远程分支对应的,一般是不可见,但是可以通过一些命令展示它们的存在:

$ git branch -r
origin/master
$ git branch -a
* master
  remotes/origin/master

总结

  • origin master 表达的意思是:git 服务器(origin 代表)上的 master 分支。

  • origin/master:本地分支,是从远程拉取代码后,在本地建立的一份拷贝。

  • git fetch origin master:从 git 服务器(origin 代表)上拉取 master 分支最新代码。

  • git merge origin/master:将当前开发分支和 origin/master 分支合并。

  • git merge origin master:将当前分支和远程 origin 默认的分支和本地的 master 分支合并。

  • git push origin master:将当前分支推送到 git 服务器(origin 代表)上的 master 分支上。

上次编辑于:
贡献者: rumosky