merge origin master 和 merge origin/master的不同之处
说明
在网上找到的普遍解释如下:
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 分支上。