1、Commit对象介绍

现在来介绍最后一种Git对象commit对象,也叫提交对象。

提交对象可以理解为是对树对象的一层封装,提交信息包括基于当前暂存区中索引文件生成的tree对象,还有包含了提交时间,提交者信息,作者信息,以及提交备注等内容,更重要的是里面还包含了父提交的ID,由此就可以形成Git提交的有向无环图。(是链式的关系,把所有commit对象关联起来)

即:commit对象通常指向一个 tree 对象,并且封装了文件的提交时间,提交者信息,作者信息,提交备注,以及父提交引用等数据。

下面是commit对象的存储结构:

2、Commit对象说明

我们通过练习来说明commit对象,接着用前面Tree对象的本地版本库。

(1)创建一个commit对象

我们可以通过调用commit-tree命令创建一个提交对象,为此需要指定一个树对象的SHA-1值,以及该提交的父提交对象。

说明:使用commit-tree命令来创建提交对象,一般都需要和父提交进行关联,如果是第一次将暂存区的文件索引数据提交到本地版本库,那么该提交操作就不需要指定父提交对象。

1)我们可以先查看一下此时Git本地库中的对象,如下

.git/objects/01/ab2a43b1eb150bcf00f375800727df240cf653 # 第三个tree树对象
.git/objects/0c/1e7391ca4e59584f8b773ecdbbb9467eba1547 # test.txt第二个版本(blob对象)
.git/objects/16/3b45f0a0925b0655da232ea8a4188ccec615f5 # 第二个tree树对象
.git/objects/83/baae61804e65cc73a7201a7252750c76066a30 # test.txt第一个版本(blob对象)
.git/objects/d8/329fc1cc938780ffdd9f94e0d364e0ea74f579 # 第一个tree树对象
.git/objects/fa/49b077972391ad58037050f2a75f74e3671e92 # new.txt第一个版本(blob对象)

2)我们通过第一个树对象,创建一个commit对象

# 1.做提交操作,创建一个commit对象
L@DESKTOP-T2AI2SU MINGW64 /j/git-repository/git_learning (master)
$ echo 'first commit' | git commit-tree d8329f
3ceba95d3cd9cce982d31e41e3b995ece72f755d # 2.确定该对象类型
L@DESKTOP-T2AI2SU MINGW64 /j/git-repository/git_learning (master)
$ git cat-file -t 3ceba95d3c
commit # 3.查看该对象内容
L@DESKTOP-T2AI2SU MINGW64 /j/git-repository/git_learning (master)
$ git cat-file -p 3ceba95d3c
tree d8329fc1cc938780ffdd9f94e0d364e0ea74f579
author sun_wk <sun_wk@126.com> 1618190880 +0800
committer sun_wk <sun_wk@126.com> 1618190880 +0800 first commit

说明:

  • tree:表示该commit对象所指向的tree对象的索引
  • author:表示该文件的作者。
  • committer:表示该文件的提交者。
  • first commit:这段文本是提交备注。(备注与前面留空一行)
  • 因为是第一次进行commit提交操作,所以没有父提交信息。
  • 1618190880 +0800:表示时间,一个时间戳。

即:commit对象的格式很简单:指明了该时间点项目快照的顶层树对象、作者/提交者信息(从 Git 设置的 user.nameuser.email中获得),以及当前时间戳、留空一行,最后是提交注释。

提示:git commit-tree命令不但生成了提交对象,而且会将对应的快照(树对象)提交到本地库中。

(2)创建第二个commit对象

根据第二个tree对象和第一个commit对象,来创建第二个commit对象。

通过-p选项指定父提交对象。

# 1.创建第二个commit对象
L@DESKTOP-T2AI2SU MINGW64 /j/git-repository/git_learning (master)
$ echo 'second commit' | git commit-tree 163b45f0a09 -p 3ceba95d3cd9cc
60e1c209e9de87314ec47cf28e61de8df5362fe6 # 2.查看该对象内容
L@DESKTOP-T2AI2SU MINGW64 /j/git-repository/git_learning (master)
$ git cat-file -p 60e1c209e9de8
tree 163b45f0a0925b0655da232ea8a4188ccec615f5
parent 3ceba95d3cd9cce982d31e41e3b995ece72f755d
author sun_wk <sun_wk@126.com> 1618193286 +0800
committer sun_wk <sun_wk@126.com> 1618193286 +0800 second commit

提交对象的格式很简单:

它先指定一个顶层树对象,代表当前项目快照;

然后是可能存在的父提交;

之后是作者/提交者信息(依据你的 user.nameuser.email 配置来设定,外加一个时间戳);

留空一行,最后是提交注释。

第三个commit提交,同上,这里就不演示了。

3、本地库中对象之间的关系

我们可以查看一下此时Git本地库中的对象

.git/objects/01/ab2a43b1eb150bcf00f375800727df240cf653 # 第三个tree树对象
.git/objects/0c/1e7391ca4e59584f8b773ecdbbb9467eba1547 # test.txt第二个版本(blob对象)
.git/objects/16/3b45f0a0925b0655da232ea8a4188ccec615f5 # 第二个tree树对象
.git/objects/3c/eba95d3cd9cce982d31e41e3b995ece72f755d # 第一个commit提交对象
.git/objects/46/ab608799a0e65e970b67b9b52f6c1407c39036 # 第三个commit提交对象
.git/objects/60/e1c209e9de87314ec47cf28e61de8df5362fe6 # 第二个commit提交对象
.git/objects/83/baae61804e65cc73a7201a7252750c76066a30 # test.txt第一个版本(blob对象)
.git/objects/d8/329fc1cc938780ffdd9f94e0d364e0ea74f579 # 第一个tree树对象
.git/objects/fa/49b077972391ad58037050f2a75f74e3671e92 # new.txt第一个版本(blob对象)

可以从上面看到,此时的本地版本库中共有9个对象,三个blob对象,三个tree对象,三个commit对象。

他们之间的关系如下图:

4、总结

  1. 提交是我们经常使用的Git动作,每次提交操作都指向一个树对象,同时会产生一个commit对象。

    即:一个commit对象包含了一个tree对象,这个tree对象记录了在那个时间点,项目包含了什么文件夹和什么文件。
  2. 一个提交对象可以有一个或者多个父提交。
  3. 每次commit操作都会基于当前索引文件index新建tree对象。那么当前索引文件,是在上次提交的基础上更新来的,所以每次提交产生的commit对象,与其他的commit对象,都有前后关系或者称为父子关系。
  4. 对于我们来说,不需要直接访问blob对象和tree对象,我们直接访问commit对象就可以了。

    即:commit对象对应的tree对象下面,又包含了小的tree对象和blob对象,子的tree对象一层层展开,最后叶子节点就是一个个blob对象,也就是一个个文件。

到这里,我们就能够清楚的了解,什么叫一个Git版本。tree对象才是一次项目版本的快照,提交对象是对tree对象的一次封装。

即:

  • 项目的快照就是一个树对象。
  • 项目的版本就是一个提交对象。

而且Git的每一个版本,存储的不是增量,而存储的是当前项目的快照。同时objects目录中相当于存放了项目的所有历史记录,回滚就相当的方便了,找到对应的commit对象的hash就可以了。

5、练习

请问下图中包含多少个tree对象和blob对象?

一共包含两个tree对象,一个blob对象,一个commit对象。

说明:

  • 一个commit对象一定对应一个tree对象(这个tree对象应该是一个完整项目仓库的快照)
  • doc目录下有一个blob对象,也就是readme文件。

6、本文用到的命令总结

Git底层命令:

  • git commit-tree:生成一个commit对象。
  • git cat-file -t 键:查看Git对象的类型。
  • git cat-file -p 键:查看Git对象的内容。

参考:

『现学现忘』Git对象 — 17、Commit对象的更多相关文章

  1. 『现学现忘』Git基础 — 18、Git对象的总结

    目录 1.Git操作最基本的流程 2.工作目录中文件的状态 3.Git效率说明 提示:前面三篇文章已经分别的对blob对象.tree对象.commit对象进行了详细的说明,这篇文章我们总结一下,Git ...

  2. 『现学现忘』Git对象 — 16、Tree对象详解

    目录 1.Tree对象介绍 2.Tree对象说明 (1)初始化一个新的本地版本库 (2)创建一个树对象(重点) (3)创建第二个文件(重点) (4)将第一个树对象加入暂存区,使其成为新的树对 3.总结 ...

  3. 『现学现忘』Git分支 — 39、Git中分支与对象的关系

    目录 1.Git对象之间的关系 2.提交对象与分支的关系 (1)提交对象与分支的关系 (2)分支说明 (3)HEAD与分支的关系 1.Git对象之间的关系 我们之前学了Git的三个对象:提交对象.树对 ...

  4. 『现学现忘』Git对象 — 15、blob对象介绍

    目录 (一)Git对象的存放目录 (二)Git中对象类型 (三)blob对象 1.blob对象说明 (1)blob对象定义 (2)blob对象说明 (3)blob对象存储的方式 (4)查看blob对象 ...

  5. 『现学现忘』Git基础 — 5、Git的协作模式

    目录 1.分布式工作流程 2.集中式工作流 3.分支工作流 4.GitFlow 工作流(最流行) 5.Forking 工作流(偶尔使用) 6.总结 1.分布式工作流程 与传统的集中式版本控制系统(CV ...

  6. 『现学现忘』Git基础 — 36、标签tag(一)

    目录 1.标签介绍 2.列出标签 3.创建标签 (1)标签的分类 (2)附注标签 (3)轻量标签 4.后期打标签 1.标签介绍 软件的某个发行版本所对应的,其实就是软件开发过程中,某一个阶段的最后一次 ...

  7. 『现学现忘』Git基础 — 37、标签tag(二)

    目录 5.共享标签 6.删除标签 7.修改标签指定提交的代码 8.标签在.git目录中的位置 9.本文中所使用到的命令 提示:接上一篇文章内容. 5.共享标签 默认情况下,git push 命令并不会 ...

  8. 『现学现忘』Git基础 — 24、Git中查看历史版本记录

    目录 1.查看详细的历史版本记录 2.简化显示历史版本记录 3.历史版本记录常用操作 (1)指定查看最近几次提交的内容 (2)以简单图形的方式查看分支版本历史 (3)翻页与退出 4.查看分支相关的版本 ...

  9. 『现学现忘』Git后悔药 — 33、revert撤销(二)

    目录 4.一次移除某几次提交 (1)git revert移除某几次提交的修改 (2)git revert 移除某几次连续的提交的修改 5.revert命令常用参数 6.git revert和git r ...

随机推荐

  1. Ubuntu16.04 搭建samba服务器

    1昨天花了一天时间弄了NFS服务器,结果搭建完之后出现各种问题,要么挂载不上,要么就是字符乱码.今天在看到一个关于树莓派的介绍的时候,提到Samba服务器的搭建,我尝试了一下,结果发现很顺利地就能够正 ...

  2. 网络编程 --安装wkhtmltopdf出现中文乱码的情况

    1 首先下载安装包 2安装依赖文件apt-get install libxfont1 xfonts-encodings xfonts-utils xfonts-base xfonts-75dpi su ...

  3. python实现AES加密、解密

    AES加密方式有五种:ECB, CBC, CTR, CFB, OFB 从安全性角度推荐CBC加密方法,本文介绍了CBC,ECB两种加密方法的python实现 python 在 Windows下使用AE ...

  4. MySQL_fetch_array 和 MySQL_fetch_object 的区别是 什么?

    以下是 MySQL_fetch_array 和 MySQL_fetch_object 的区别: MySQL_fetch_array() – 将结果行作为关联数组或来自数据库的常规数组返回. MySQL ...

  5. C++ | 虚表的写入时机

    虚表 在C++的多态机制中,使用了 virtual 关键字声明的函数称之为虚函数,每个有虚函数的类或者虚继承的子类,编译器都会为它生成一个虚拟函数表(简称:虚表,以下用 vftable表示),表中的每 ...

  6. 【HTML5版】导出Table数据并保存为Excel

    首发我的博客 http://blog.meathill.com/tech/js/export-table-data-into-a-excel-file.html 最近接到这么个需求,要把<tab ...

  7. 手把手教你从零写一个简单的 VUE

    本系列是一个教程,下面贴下目录~1.手把手教你从零写一个简单的 VUE2.手把手教你从零写一个简单的 VUE--模板篇 今天给大家带来的是实现一个简单的类似 VUE 一样的前端框架,VUE 框架现在应 ...

  8. Codepen 每日精选(2018-4-28)

    按下右侧的"点击预览"按钮可以在当前页面预览,点击链接可以打开原始页面. 页面目录特效https://codepen.io/suez/pen/k... 选单交互效果https:// ...

  9. spark-shell报错java.lang.IllegalArgumentException: java.net.UnknownHostException: namenode

    在使用spark on yarn启动spark-shell时,发现报错: 是说找不到主机名为namenode的主机,那么应该是配置文件出错了. 经过检查,发现是spark-defaults.conf文 ...

  10. Android Studio安装问题

    安装问题可以参考:https://blog.csdn.net/y74364/article/details/96121530 但是gradle安装缓慢,需要FQ.有加速器FQ的可以开加速器安装,没有的 ...