说明

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

  1. git merge origin master
  2. # 将origin merge 到 master 上
  3. git merge origin/master
  4. # 将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分支:

  1. git fetch origin master
  2. git merge origin branchB

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

  1. git merge origin/master branchB

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

  1. 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,例如:

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

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

  1. $ git remote
  2. origin

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

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

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

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

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

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

origin/master的由来

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

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

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

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

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

  1. $ git branch -r
  2. origin/master
  3. $ git branch -a
  4. * master
  5. 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分支上。