Git submodule的坑
by
at 2013-05-08 21:44:00
original http://blog.devtang.com/blog/2013/05/08/git-submodule-issues/
前言
随着近几年的发展,Git已经成为开源界的标准的版本控制工具。开源界的重量级项目,如Linux, Android, Eclipse, Gnome, KDE, Qt, ROR, Debian,无一例外的都是使用git来进行版本控制。如果你还不会Git,那么恕我直言,你已经out了,赶紧抽空充充电吧。本文并不打算做Git入门级介绍,想学习git的同学,推荐国内作者蒋鑫写的《Git权威指南》。
对于一些比较大的工程,为了便于复用,常常需要抽取子项目。例如我开发的猿题库客户端现在包括3门考试,客户端涉及的公共UI、公共底层逻辑、公共的第三方库、以及公共的答题卡扫描算法就被我分别抽取成了子项目。这些子项目都以git submodule的形式,增加到工程中。
在使用了git submodule一段时间后,我发现了一些submodule的问题,在此分享给大家。
更新submodule的坑
submodule项目和它的父项目本质上是2个独立的git仓库。只是父项目存储了它依赖的submodule项目的版本号信息而已。如果你的同事更新了submodule,然后更新了父项目中依赖的版本号。你需要在git pull之后,调用 git submodule update来更新submodule信息。
这儿的坑在于,如果你git pull之后,忘记了调用 git submodule update,那么你极有可能再次把旧的submodule依赖信息提交上去。对于那些习惯使用 git commit -a的人来说,这种危险会更大一些。所以建议大家:
- git pull之后,立即执行git status, 如果发现submodule有修改,立即执行git submodule update
- 尽量不要使用 git commit -a, git add命令存在的意义就是让你对加入暂存区的文件做二次确认,而 git commit -a相当于跳过了这个确认过程。
更复杂一些,如果你的submodule又依赖了submodule,那么很可能你需要在git pull 和 git submodule update之后,再分别到每个submodule中再执行一次git submodule update,这里可以使用 git submodule foreach命令来实现: git submodule foreach git submodule update
修改submodule的坑
有些时候你需要对submodule做一些修改,很常见的做法就是切到submodule的目录,然后做修改,然后commit和push。
这里的坑在于,默认git submodule update并不会将submodule切到任何branch,所以,默认下submodule的HEAD是处于游离状态的(‘detached HEAD’ state)。所以在修改前,记得一定要用git checkout master将当前的submodule分支切换到master,然后才能做修改和提交。
如果你不慎忘记切换到master分支,又做了提交,可以用cherry-pick命令挽救。具体做法如下:
- 用
git checkout master
将HEAD从游离状态切换到 master 分支, 这时候,git会报Warning说有一个提交没有在branch上,记住这个提交的change-id(假如change-id为 aaaa) - 用
git cherry-pick aaaa
来将刚刚的提交作用在master分支上 - 用
git push
将更新提交到远程版本库中
以下是相关命令的操作示范和命令行输出结果:
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
使用第三方工具
对于submodule的重度使用者,有几个工具可作推荐:
- Repo Google用于管理Android项目的工具。
- Gitslave
- Git Subtree
以上工具,我都没有实际用过,所以无法提供更多信息。
Tips
由于常常使用submodule的相关命令,可以在 ~/.gitconfig文件中将其设置别名,方便操作,我设置的所有相关别名如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|