分支
分支是什么
- Git的分支其实就是一个指针,它指向的是某个提交对象。
- Git在提交时就会保存一个提交对象。
- 提交对象包含一个指向暂存内容快照的指针,作者姓名邮箱,提交说明,指向父对象的指针。
之前说过,Git以类似文件系统的方式管理文件,保存一系列不同时刻的文件快照。
执行暂存操作时,Git会计算相应文件的校验和,把当前版本的文件快照保存到Git仓库(使用blob对象保存),把校验和加入暂存区。
执行提交操作时,Git会计算每一个子目录的校验和,在Git仓库中将这些校验和保存为树对象。Git随后创建一个提交对象,提交对象除了指向暂存内容快照的指针、作者信息、提交说明、父对象指针外,还包含指向树对象的指针。那么Git就记录了此次提交的全部信息。
首次提交后,Git仓库有三类对象:
- blob对象:保存文件快照
- 树对象:保存目录结构,blob对象索引
- 提交对象:指向树对象的指针,提交信息
后续提交后,新的提交对象会包含指向上次提交对象(即父对象)的指针:
分支——本质上仅仅是指向提交对象的可变指针。切换不同的分支,其实就是切换不同的提交对象,重现该提交对象的快照。
git init
初始化一个项目时,默认创建了一个*master
*分支,master
分支与其他分支没有任何区别。每次提交,master
就向前移动。
创建分支——git branch
创建分支其实就是创建一个新的可变指针,指向了当前的提交对象。
除了默认的master
分支,Git还有一个HEAD
指针,指向当前所在的本地分支
$ git branch testing 仅创建分支
$ git log --oneline --decorate
f30ab (HEAD -> master, testing) blabla
34ac2 blabla
98ca9 blabla
此时,当前所在的本地分支是master
分支,master
分支指向的提交对象是f30ab
,testing
分支是基于当前的提交对象f30ab
创建的,所以testing
也指向了提交对象f30ab
。由于只创建而没有切换到testing
分支,所以HEAD
仍指向了master
分支。
切换分支——git checkout
$ git checkout testing 切换到已存在的分支testing
$ git checkout -b testing 创建并切换到新分支testing
切换之后,分支指向如图:
如图,master
和testing
仍指向提交对象30ab
,但是HEAD
指向了testing
,即当前分支已经被切换到了testing
。
如果此时进行提交,创建了一个提交对象87ab2
,那么HEAD
和testing
都会向前移动,指向新的提交对象87ab2
,而master
仍然指向f30ab
(没有基于master
分支提交),结果如图:
如果此时再切换回master
分支:git checkout master
,会把HEAD
指回master
,同时把工作区的内容恢复成master
指向的快照,即提交对象f30ab
的内容:
此时,工作区已经处于master
分支,工作区是f30ab
的快照内容。
此时再次提交,创建提交对象c2b9e
,它的父对象就是f30ab
,master
和HEAD
向前移动。之前在testing
分支的提交对象87ab2
的父对象也是f30ab
,项目从f30ab
起,开始分叉:
$ git log --oneline --decorate --graph --all 输出提交历史、各个分支指向、项目的分支分叉
* c2b9e (HEAD, master) made other changes
| * 87ab2 (testing) made a change
|/
* f30ab add feature #32 - ability to add new formats to the
* 34ac2 fixed bug #1328 - stack overflow under certain conditions
* 98ca9 initial commit of my project