GIT是非常优秀的版本控制工具,但是苦于git那晦涩难懂的man pages,还有众多的命令选项和怪异的用法,git有点难学。这篇文章分享我学习过程中收藏的一些好图,并围绕这些图讲讲我对git的理解,希望对大家有所帮助。
GIT工作流程
了解git,首先要弄清楚对象在被git管理过程中所处的4个阶段,分别是:工作目录、index(又称为暂存区)、本地仓库和远程仓库。一开始接触这些概念可能比较绕,其实在git入门阶段,可以先抛开远程仓库不看,只了解管理本地仓库的”3棵树”就够了。如下图:
常用GIT命令
在开始之前,我们需要把下面的图看懂:
HEAD,头,它始终指向当前所处分支的最新的提交点。你所处的分支变化了,或者产生了新的提交点,HEAD就会跟着改变。
working directory,它是你的工作目录,也是当前你看到的东西。你的工作目录是与版本、分支相关的。
stage的东西虽然看不见,但是执行git status就会看到哪些对象的修改将在下一次commit的时候被放进本地仓库。这些东西称为stage。
commit
commit把暂存区的内容存入到本地仓库,并使得当前分支的HEAD向后移动一个提交点。如果对最后一次commit不满意,可以使用git commit --amend来进行撤销,修改之后再提交。如图所示的,ed489被4ca87取代,但是git log里看不到ed489的影子,这也正是amend的本意:原地修改,让上一次提交不露痕迹。
checkout
checkout用来检出并切换分支。checkout成功后,HEAD会指向被检出分支的最后一次提交点。对应的,工作目录、暂存区也都会与当前的分支进行匹配。下图是执行git checkout maint后的结果:
reset
reset命令把当前分支指向另一个位置,并且相应的变动工作目录和索引。如下图,执行git reset HEAD~3后,当前分支相当于回滚了3个提交点,由ed489回到了b325c:
reset有3种常用的模式:
–soft,只改变提交点,暂存区和工作目录的内容都不改变
–mixed,改变提交点,同时改变暂存区的内容。这是默认的回滚方式
–hard,暂存区、工作目录的内容都会被修改到与提交点完全一致的状态
diff
我们在commit、merge、rebase、打patch之前,通常都需要看看这次提交都干了些什么,于是diff命令就派上用场了:
来比较下上图中5种不同的diff方式:
比较不同的提交点之间的异同,用git diff 提交点1 提交点2
比较当前分支与其他分支的异同,用git diff 其他分支名称
在当前分支内部进行比较,比较最新提交点与当前工作目录,用git diff HEAD
在当前分支内部进行比较,比较最新提交点与暂存区的内容,用git diff --cached
在当前分支内部进行比较,比较暂存区与当前工作目录,用git diff
看起来有点复杂?是的,记不住的时候就看看这些图吧。
merge
merge命令把不同的分支合并起来。如下图,HEAD处于master分支的ed489提交点上,other分支处于33104提交点上,项目负责人看了下觉得other分支的代码写的不错,于是想把代码合并到master分支,因此直接执行git merge other,如果没有发生冲突,other就成功合并到master分支了。
rebase
rebase又称为衍合,是合并的另外一种选择。merge把两个分支合并到一起进行提交,无论从它们公共的父节点开始(如上图,other分支与master分支公共的父节点b325c),被合并的分支(other分支)发生过多少次提交,合并都只会在当前的分支上产生一次提交日志,如上图的f8bc5。所以merge产生的提交日志不是线性的,万一某天需要回滚,就只能把merge整体回滚。而rebase可以理解为verbosely merge,完全重演下图分支topic的演化过程到master分支上。如下图:
原文转自:http://nettedfish.sinaapp.com/blog/2013/08/05/deep-into-git-with-diagrams/