【Git教程】(一)基本概念 ——工作流、分布式版本控制、版本库 ~

Git教程 · 基本概念

  • 1️⃣ 为什么要用 Git
  • 2️⃣ 为什么要用工作流
  • 3️⃣ 分布式版本控制
  • 4️⃣ 版本库
  • 5️⃣ 简单的分支创建与合并
  • 🌾 总结

在本章中,将介绍一个分布式版本控制系统的设计思路,以及它与集中式版本控制系统的不同之处。除此之外,还将带你了解分布式版本库的具体工作方式,以及为什么我们会说,在Git 中创建分支和合并分支不是个大不了的问题。

在这里插入图片描述


1️⃣ 为什么要用 Git

Git的背后有着一个非常精彩的成功故事。2005年4月,Linus Torvalds 因不满当时任何
一个可用的开源版本控制系统,就亲自着手实现了Git。
时至今日,如果我们在 Google 中搜索 “git version control” 这几个关键词,都会看到数以百万计的返回结果。Git已经俨然成为了新型开源项目的一个标准。许多大型的开源项目都已经或正在计划迁移到Git 上来。

下面,我们来看一下这么多人之所以会选择 Git 的原因。

  • Git 允许我们利用分支来开展工作:在一个由多个开发者并行协作的项目中,开发者各自会有很多不同的开发路线。Git 的优势在于,它提供了一整套针对开发链的重新整合工具,以便我们对其进行合并、变基和捡取等操作。
  • 工作流上的灵活性:Git 非常灵活。不但单一开发者可以用它,敏捷团队也可以找到使用它工作的合适方法,甚至一个由众多开发者在不同的工作地点参与的大型国际项目也可以用它开发出一个很好的工作流。
  • 适合奉献合作:大多数开源项目所依靠的都是开发者的无私奉献。因此,让这种无私 奉献的方式尽可能地简单化是一件非常重要的事。而这在一个集中式的版本控制系统中通常是很难做到的,因为我们不可能让所有人都有权限去写版本库。但如果我们使用 Git, 那么每个人都先可以克隆一个独立的工作版本库,然后再对其进行后续的改动。
  • 高性能:Git在处理拥有许多文件且历史悠久的项目时速度也依然是非常快的。例如,使用 Git 将 Linux 内核源码的当前版本切换到6年前的旧版本时,在一台 MacBook Air上所需的时间不到1分钟。考虑这两个版本之间有着超过200000次的提交和40000个更改文件,这已经足以让人印象深刻了。
  • 强大的抗故障和抗攻击能力:由于项目历史被分散存储在多个分布式版本库中,因此 数据严重流失的可能性不大。再加上版本库中有着巧妙简单的数据结构,这确保了其中的数据即使在遥远的未来也仍然会被正确地解释。而且,它还使用了统一的加密校验,这使得攻击者难以对版本库进行篡改。
  • 离线开发与多点开发:分布式的体系结构可以使得离线开发或者边旅行边开发的方式变得非常容易。而且该结构在多点开发模式下,我们既不需要设置中央服务器,也不需要固定的网络连接。
  • 强大的开源社区:除官方提供的详细文档外,你还可以在该社区找到无数相关的手册、 论坛、维基网站等,另外还有各种工具生态系统、托管平台、出版物、服务以及针对各个开发环境的插件,整个社区都正在茁壮成长。
  • 可扩展性:Git 为用户提供了许多实用命令,其中包括了能使我们更便于直接访问其远程版本库的命令。这可以让Git 变得非常灵活,这种灵活性将允许其各种独立应用提供比默认的Git 版本更为强大的功能。

2️⃣ 为什么要用工作流

Git 非常灵活。可为多种不同的角色所用,从偶尔需要版本化少量 shell 脚本的单一系统 管理员,到 Linux 内核项目中的上百个开发人员,一切皆有可能。当然,这种灵活性不是没有代价的。在开始用Git来开展工作之前,你还必须要做一组决定。例如以下几种。

  • Git 中固然已经是分布式版本库。但你是真的打算只在本地工作,还是更愿意建立一个中央版本库?
  • Git 支持 push 和 pull 两种数据传输类型,但我们需要同时使用它们吗? 如果让你选, 你会选哪一个? 为什么不是另一个?
  • 分支与合并是 Git中两个强大的功能。但是,我们应该开多少个分支呢? 是根据每个软件功能来开? 还是针对每个发行版来开?还是只该有一个分支?

为了便于入门,下面我们来总结一下工作流及其作用。

  • 工作流指的是相关项目的日常操作规程。
  • 工作流会给出具体的步骤。
  • 工作流会显示必要的命令和选项。
  • 工作流非常适用于密切的团队合作,而目前的这些现代软件项目通常就出自这样的合作。
  • 一些工作流可能并不是目标问题唯一正确的解决方案,但它们是一个很好的起点,我们 可以从中为自己的项目开发出高效的工作流。

我们之所以会重点介绍商业项目中敏捷开发团队的工作,是因为我们相信目前许多专业 开发者(包括作者)都处于这样的工作环境中。当然,这里并不包括那些具有特殊要求的大 型项目,因为这些项目通常有着很夸张的工作流,而且我们相信这些也不是大多数开发者会 感兴趣的项目。另外,这里也不包括那些开源项目的开发,虽然这些项目也可以用 Git 规划出一个很有意思的工作流。


3️⃣ 分布式版本控制

在具体探讨分布式版本控制的概念之前,让我们先来快速回顾一下传统的集中式版本控制架构。
下图中所显示的就是一个集中式版本控制系统(例如 CVS 或 Subversion) 的典型布局。 每个开发者都在他或她自己的计算机上有一个包含所有项目文件的工作目录(即工作区)。当该开发者在本地做了修改之后,他或她就会定期将修改提交给某台中央服务器。然后,开发者在执行更新操作的同时也会从该服务器上捡取出其他开发者所做的修改。这台中央服务器上存储着这些文件(即版本库)的当前版本和历史版本。因此,这些被并行开发的分支,以及各种被命名(标记)的版本都将会被集中管理。


在这里插入图片描述

而在分布式版本控制系统(见上图)中,开发者环境与服务器环境之间是没有分隔的。 每一个开发者都同时拥有一个用于当前文件操作的工作区与一个用于存储该项目所有版本、 分支以及标签的本地版本库(我们称其为一份克隆)。每个开发者的修改都会被载入成一次次 的新版本提交(commit), 首先提交到其本地版本库中。然后,其他开发者就会立即看到新的版本。通过推送 (push) 和拉回 (pull) 命令,我们可以将这些修改从一个版本库传送到另一个 版本库中。这样一来,从技术上来看,这里所有的版本库在分布式架构上的地位是同等的。 因此从理论上来讲,我们不再需要借助服务器,就可以将某一台开发工作机上所做的所有修改直接传送给另一开发工作机。当然在具体实践中,Git 中的服务器版本库也扮演了重要的角色,例如以下这些特型版本库。

  • 项目版本库(blessed repository): 该版本库主要用于存储由“官方”创建并发行的版本。
  • 共享版本库(shared repository): 该版本库主要用于开发团队内人员之间的文件交换。 在小型项目中,项目版本库本身就可以胜任这一角色了。但在多点开发的条件下,我 们可能就会需要几个这样的专用版本库。
  • 工作流版本库(workflow repository): 工作流版本库通常只用于填充那些代表工作流 中某种特定进展状态的修改,例如审核通过后的状态等。
  • 派生版本库 ( fork repository):该版本库主要用于从开发主线分离出某部分内容(例如,分离出那些开发耗时较长,不适合在一个普通发布周期中完成的内容),或者隔离出可能永远不会被包含在主线中的、用于实验的那部分开发进展。

下面,我们再来看看分布式系统相对于集中式的优点有哪些。

  • 高性能:几乎所有的操作都无需进行网络访问,均可直接在本地执行。
  • 高效的工作方式:开发者可通过多个本地分支在不同任务之间进行快速切换。
  • 离线功能:开发者可以在没有服务器连接的情况下执行提交、创建分支、版本标签等 操作。之后再将其上传服务器。
  • 灵活的开发进程:我们可以在团队和公司中为其他部门建立专用的版本库,例如为方 便与测试人员交流而建的版本库。这样相关修改就很容易发布,因为只是特定版本库 上的一次推送。
  • 备份作用:由于每个开发者都持有一份拥有完整历史版本的版本库副本,所以因服务器故障而导致数据丢失的可能性是微乎其微的。
  • 可维护性:对于那些难以对付的重构工作,我们可以在将成功传送给其原始版本库之前,先在该版本库的副本上尝试一下。

4️⃣ 版本库

其实,版本库本质上就是一个高效的数据存储结构而已,由以下部分组成。

  • 文件(即 blob): 这里既包含了文本也包含了二进制数据,这些数据将不以文件名的 形式被保存。
  • 目录(即 Tree): 目录中保存的是与文件名相关联的内容,其中也会包含其他目录。
  • 版本(即 commit): 每一个版本所定义的都是相应目录的某个可恢复的状态。每当我们创建一个新的版本时,其作者、时间、注释以及其之前的版本都将会被保存下来。

对于所有的数据,它们都会被计算成一个十六进制散列值(例如像1632acb65b01c6b621d6e1105205773931bbla41 这样的值)。这个散列值将会被用作相关对象的引用,以及日后恢复数据时所需的键值(见下图)。


在这里插入图片描述

也就是说, 一个提交对象的散列值实际上就是它的“版本号”,如果我们持有某一提交的 散列值,就可以用它来检查对应版本是否存在于某一版本库中。如果存在,我们就可以将其 恢复到当前工作区相应的目录中。如果该版本不存在,我们也可以从其他版本库中单独导入(拉回)该提交所引用的全部对象。

接下来,我们来看看采用这种散列值和这种既定的版本库结构究竟有哪些优势。

  • 高性能:通过散列值来访问数据是非常快的。
  • 冗余度——释放存储空间:相同的文件内容只需存储一次即可。
  • 分布式版本号:由于相关散列值是根据文件,作者和日期来计算的,所以版本也可以 “离线”产生,不用担心将来会因此而发生版本冲突。
  • 版本库间的高效同步:当我们将某一提交从一个版本库传递给另一个版本库时,只需要传送那些目标版本库中不存在的对象即可。而正是因为有了散列值的帮助,我们才能很快地判断相关对象是否已经存在。
  • 数据完整性:由于散列值是根据数据的内容来计算的,所以我们可以随时通过Git 来查看某一散列值是否与相关数据匹配。以检测该数据上可能的意外变化或恶意操作。
  • 自动重命名检测: 被重命名的文件可以被自动检测到,因为根据该文件内容计算出的 散列值并没有发生变化。也正因为如此, Git 中并没有专用的重命名命令,只需移动命令即可。

5️⃣ 简单的分支创建与合并

对于大多数版本控制系统来说,分支的创建与合并通常会因其特殊性而被认为是高级拓展操作。但由于Git 最初就是为了方便那些散落在世界各地的Linux 内核开发者而创建的,合并多方努力的结果一直都是其面临的最大挑战之一 ,所以 Git 的设计目标之一就是要让分支的创建与合并操作变得尽可能地简单且安全。

在下面的图中,展示了开发者是如何通过创建分支的方式来进行并行开发 的。图中的每一个点都代表了该项目的一个版本(即commit) 。 而由于在Git 中,我们只能对整个项目进行版本化,所以每个点同时也代表了属于同一版本的各个文件。


在这里插入图片描述
如上所示,图中两位开发者的起点是同一个版本。之后两人各自做了修改,并提交了修 改。这时候,对于这两位开发者各自的版本库来说,该项目已经有了两个不同的版本。也就 是说,他们在这里创建了两个分支。接下来,如果其中一个开发者想要导入另一个人的修改, 他/她就可以用Git 来进行版本合并。如果合并成功了,Git 就会创建一个合并提交,其中会包含两位开发者所做的修改。这时如果另一位开发者也取回了这一提交,两位开发者的项目就又回到了同一个版本。

在上面的例子中,分支的创建是非计划性的,其原因仅仅是两个开发者在并行开发同一个软件罢了。在 Git 中,我们当然也可以开启有针对性的分支,即显式地创建一个分支(见下图)。显式分支通常主要用于协调某一种功能性的并行开发。


在这里插入图片描述

版本库在执行拉回和推送操作时,可以具体指定其针对的是哪一些分支。当然,除了这些简单的分支创建和合并处理外,我们也可以对分支执行以下动作。

  • 移植分支:我们可以直接将某一分支中的提交转移到另一个版本库中。
  • 只传送特定修改:我们可以将某一分支中的某一次或某几次提交直接复制到另一个分 支中。这就是所谓的捡取处理。
  • 清理历史:我们可以对分支历史进行改造、排序和删除。这有利于为该项目建立更好 的历史文档。我们称这种处理为交互式重订 (interactive rebasing)。

🌾 总结

在阅读完本文之后,你现在基本上熟悉了 Git 中的这些基本概念。也就是说, 即使你现在放下了这本书, 你也可以参加与分布式版本控制系统有关的讨论,阐述其中使用散列值的必要性和实用性,介绍 Git中的分支创建与合并操作了。

当然,你可能还会有以下疑问。

  • 我们应该如何利用这些基本概念来管理项目呢?
  • 我们应该如何协调多个版本库呢?
  • 我们究竟需要多少分支呢?
  • 我们应该如何整合自己的构建服务器呢?

对于第一个问题,可以继续阅读下一章内容。在下一章中,你将会看到那些具体用于创建版本库、版本以及版本库之间更替提交的命令。另外,如果你是一个繁忙的项目管理者,还在犹豫不决是否要采用Git 。建议再看看关于Git的局限性的的讨论,参见后续文章。



继续阅读下一篇(点击跳转)《》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/375916.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Camunda排他网关与并行网关

💖专栏简介 ✔️本专栏将从Camunda(卡蒙达) 7中的关键概念到实现中国式工作流相关功能。 ✔️文章中只包含演示核心代码及测试数据,完整代码可查看作者的开源项目snail-camunda ✔️请给snail-camunda 点颗星吧😘 💖排他网关 …

【JAVA WEB】Web标签

目录 注释标签 标题标签 h1-h6 段落标签 换行标签 格式化标签 加粗:strong 标签和 b 标签 倾斜:em 标签和 i 标签 删除线: del 标签 和 s 标签 下划线:ins 标签 和 u 标签 图片标签:img 单标签 src属性&#…

在angular12中proxy.conf.json中配置详解

一、proxy.conf.json文件的目录 二、proxy.conf.json文件中的配置 "/xxx/api": {"target": "地址/api","secure": false,"logLevel": "debug","changeOrigin": true,"pathRewrite": {"…

蓝桥杯嵌入式学习记录——点亮第一个LED(含软件的使用)

目录 一、蓝桥杯概述 二、软件的使用 三、点亮LED 一、蓝桥杯概述 蓝桥杯是一个编程大赛、商赛,获奖率高达60%(省赛中一等奖10%、二等奖20%、三等奖30%),但这并不影响它的含金量,多所高校将它列为A类赛事并实行保研…

[机器学习]K-means——聚类算法

一.K-means算法概念 二.代码实现 # 0. 引入依赖 import numpy as np import matplotlib.pyplot as plt # 画图依赖 from sklearn.datasets import make_blobs # 从sklearn中直接生成聚类数据# 1. 数据加载 # 生成(n_samples:样本点,centers&…

QT安装与helloworld

文章目录 QT安装与helloworld1.概念:2.安装QT3.配置环境变量4.创建项目5.运行效果 QT安装与helloworld 1.概念: Qt Creator是一个用于Qt开发的轻量级跨平台集成开发环境。Qt Creator可带来两大关键益处:提供首个专为支持跨平台开发而设计的…

跟着小德学C++之启动监听

嗨,大家好,我是出生在达纳苏斯的一名德鲁伊,我是要立志成为海贼王,啊不,是立志成为科学家的德鲁伊。最近,我发现我们所处的世界是一个虚拟的世界,并由此开始,我展开了对我们这个世界…

Rust开发WASM,浏览器运行WASM

首先需要安装wasm-pack cargo install wasm-pack 使用cargo创建工程 cargo new --lib mywasm 编辑Cargo.toml文件,修改lib的类型为cdylib,并且添加依赖wasm-bindgen [package] name "mywasm" version "0.1.0" edition "…

顺序图(Sequence Diagram)

也叫时序图、序列图 一、定义 顺序图是用来描述对象自身及对象间信息传递顺序的视图。 二、要素 活动者,对象,生命线,控制焦点,消息(同步消息,异步消息,返回消息,自关联消息) 1、 活动者 活动者发出情况或者接收系统的服务。 2、 对象 对象是特定行为与属性的集合。 表…

uniapp 使用renderjs引入echarts

效果图&#xff1a; 1.1renderjs引入echarts 组件zmui-echarts.vue&#xff1a; <template><view class"zmui-echarts" :prop"option" :change:prop"echarts.delay"></view> </template><script>export defaul…

互联网加竞赛 基于深度学习的行人重识别(person reid)

文章目录 0 前言1 技术背景2 技术介绍3 重识别技术实现3.1 数据集3.2 Person REID3.2.1 算法原理3.2.2 算法流程图 4 实现效果5 部分代码6 最后 0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于深度学习的行人重识别 该项目较为新颖&#xff0c;适合…

数据结构——C/栈和队列

&#x1f308;个人主页&#xff1a;慢了半拍 &#x1f525; 创作专栏&#xff1a;《史上最强算法分析》 | 《无味生》 |《史上最强C语言讲解》 | 《史上最强C练习解析》 &#x1f3c6;我的格言&#xff1a;一切只是时间问题。 ​ 1.栈 1.1栈的概念及结构 栈&#xff1a;一种特…

qt学习:mplayer播放器(视频)+arm如何播放视频实战+c启动播放器

目录 作用 linux下载 arm下载 使用方法 键盘 命令 命令词有很多&#xff0c;举例几个 在arm上qt实战 配置ui界面 添加头文件&#xff0c;成员&#xff0c;函数 添加视频按钮点击事件 列表选项双击事件 播放按钮点击事件 暂停继续按钮点击事件 停止按钮点击事件 …

挑战杯 python+深度学习+opencv实现植物识别算法系统

0 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于深度学习的植物识别算法研究与实现 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;4分工作量&#xff1a;4分创新点&#xff1a;4分 &#x1f9ff; 更多…

【Linux】缓冲区与缓冲区的刷新策略

目录 1.缓冲区基础 1.1缓冲区的刷新策略 1.1.1三种刷新策略 1.1.2.两种强制刷新策略 2.用户级语言层缓冲区 2.1.默认在显示器输出 2.2.重定向到文件输出 2.3.write调用没有显示两份的原因 3.模拟实现文件缓冲区 3.1 myFileBuffer.h 3.2 myFileBuffer.c 4.系统内核缓…

Linux系统调试课:Linux错误码介绍

文章目录 一、错误码二、错误码返回案例三、使用 goto 语句沉淀、分享、成长,让自己和他人都能有所收获!😄 📢错误代码由内核或用户空间应用程序(通过errno变量)解释。错误处理在软件开发中非常重要,而不仅仅是在内核开发中。幸运的是,内核提供的几种错误,几乎涵盖了可…

一步步建立一个C#项目(连续读取S7-1500PLC数据)

首先创建一个新项目 1、窗体应用 2、配置存储位置 3、选择框架 拖拽一个Button,可以选择视图菜单---工具箱 4、工具箱 拖拽Lable控件和TextBook控件 5、拖拽控件 接下来右键解决方案&#xff0c;点击管理NuGet程序包 6、NuGet程序包 7、安装S7net 8、点击安装 9、安装完成 …

杨幂《哈尔滨1944》现场无剧本,大胆突破令人惊艳。

♥ 为方便您进行讨论和分享&#xff0c;同时也为能带给您不一样的参与感。请您在阅读本文之前&#xff0c;点击一下“关注”&#xff0c;非常感谢您的支持&#xff01; 文 |猴哥聊娱乐 编 辑|徐 婷 校 对|侯欢庭 杨幂在《哈尔滨1944》的拍摄现场&#xff0c;竟然不带剧本&am…

空想--让MYSQL安装双引擎SQL优化器

坑人的innodb_thread_concurrencyMYSQL的优化器是硬解析, 应用每次发往MYSQL的SQL是文本格式,需要编译,虽然时间不多,也就几百毫秒的事情,可架不住SQL的请求并发量啊! 为此数据库走了两条路线, 一条是ORACLE的缓存路线, 另外一条就是MYSQL的快速路线. ORACLE是尽可能的深度…

【电路】500W 12V至220V的逆变器

本电路采用LT1073芯片DC&#xff0d;DC转换器&#xff0c;将1.5V电压升压到5V或12V。该IC有三个不同的版本&#xff0c;这取决于输出电压。两个具有5V和12V的固定输出电压&#xff0c;而最有趣的是&#xff0c;可以进行调整。该调整是通过两个电阻器的分压器&#xff0c;连接到…