Git Day.4 Branch(1)


人的目标真的是阶段性的,总是一段时间清晰的知道自己想干什么,到了下一个阶段,又会去想这么做是不是最好的选择,最开始来上海的时候,觉得只要可以写代码就很开心,到现在会有一些‘体力活’感觉难受,而且随着对项目理解的加深,会发现,现在项目里面用到的东西是很陈旧的技术,但是我人轻言微没有办法去做什么改变和调整。自己很想学新的东西,可以用于分布式管理项目的Git.还有优秀的JS框架,新的编程思想,对JS的深层理解。这些东西都在吸引我。总觉得时间不够。感叹自己浪费是时间太多,醒的有点晚,但凡事总没有早知道。

回头想想,每个人都应该越活越成熟,但是人都在慢慢变得不知道自己真正想要的是什么,因为在变成熟的过程中,对成熟的定义也一直在变。那就选一条你觉得你应该坚持下去的路,走到黑吧。虽然我已经很黑了。


学习内容:

  • 什么是分支?
  • 新建,合并分支

什么是分支?

需要知道这个问题的答案,那首先要知道Git是怎么样的一种项目管理系统,这里引用官方文档的一句话:
  "Git 和其他版本控制系统的主要差别在于,Git 只关心文件数据的整体是否发生变化,而大多数其他系统则只关心文件内容的具体差异."

如下图所示

  从图上可以看到,Git只关心那些被修改的文件,其他没有变动的文件实际都只保存了一个指向这个文件的索引(指针)
  在回忆过这一个概念之后,可以开始看看branch在Git这样的管理模式中到底扮演着什么样的角色。
  在我们执行类似于下面这样的代码时:

$ git add Readme.md
$ git commit -m "new branch"

添加文件的add指令,会对每一个文件计算校验和(即SHA-1 哈希字串),然后把当前版本的文件快照保存到 Git 仓库中(Git 使用 blob 类型的对象存储这些快照),并将校验和加入暂存区域。
  而commit指令,会产生一个commit对象,这个commit对象包括了指向,在这次提交之前,由每个子目录形成的树对象(这个树对象是为了让你在提交过这次内容之后还可以重现提交之前的状态)的指针。除此之外,commit对象中还包含本次提交的作者等相关附属信息,包含零个或多个指向该提交对象的父对象指针:首次提交是没有直接祖先的,普通提交有一个祖先,由两个或多个分支合并产生的提交则有多个祖先。
  如下图:


上面这种情况更多是对应,我们首次提交更新。
  如果我们第一次提交的基础上,进行了修改,然后准备进行第二次提交。这时候情况会跟上面的图有所不同。此时的commit对象除了包含上面所说的这些信息,还会包含一个指向前一个commit对象的parent指针,如下图:


在明白了commit对象之后,我们就很好理解分支的概念了。所谓分支就是指向一个commit对象的指针(默认这个 分支的名称是master):


这个指针是会根据你的更新一直移动的,一直指向你最新的这次更新。
  其实这很好理解,就拿自来水管道来打比方,比如现在有一根自来水管道经过A,B,C三个小区:
  A是最新通过的小区,C是最后的。
  那么如果有一天我们想新建一个线路到D小区。
  但是BD的距离比CD距离更小。
  那我们可以把这个指针指向B小区,从B小区动工然后修一条线路到D小区。

如果我们想在当前位置新修一条管道出去呢,用到指令:

$ git branch testBranch

现在我们已经修好了一条可以通道别的地方的管道, 接下来,我们得把工人都叫过来,开始在这条新的管道上工作。
  针对上面这个情况,Git里面定义了一个head指针,用来指向当前我们工作的分支管道,其实就相当于当前分支的别名。现在我们把这个head指针指向我们新建的testBranch指针

$ git checkout testBranch

这时候,我们的项目就像下面这张图一样:


然后我们在新的testBranch这个分支上新建了一个项目文件。比如说ReadMe.md

$ vim ReadMe.md  #新建ReadMe.md文件
$ git commit -a -m "Created the ReadMe.md " #提交我们刚刚新建的文档到testBranch分支

这样操作结果如下:

我们可以很清楚的看到,这是我们的testBranch分支已经往最前的位置移动,如果此时我们再切换到master分支,同样也新建一个文件。那就会产生一个分叉。

$ git checkout  master
$ vim ReadMeCopy.md
$ git commit -a -m "Created the ReadMeCopy.md"

结果如图:

关于第一小段"什么是分支?"官方的例子到这里结束了。那么,分支到底是什么,跟其他的版本控制工具,到底有什么区别?

分支的实际内容,是一段地址。地址指向一个commit对象,而commit对象中包含的是:
  1. 指向 在这次提交之前,由每个子目录形成的树对象 的指针。
  2. 本次提交的作者等相关附属信息。
  3. 包含零个或多个指向该提交对象的父对象指针。(第一次提交是没有的)

跟其他的版本控制工具的区别在哪呢?(个人见解)
  1. 在Git中,我们可以很轻易的回到之前的历史版本,我们只需要切换分支,实际上也就是把head指针移动到其他的分支上。
  而其他类型的版本控制工具,会把整个分支的文件备份到不同的目录下。如果我们要切换过去,就必须把这个分支上的所有文件同步到本地。
  2. 团队合作,我们可以让不同的同事,在同一个parent分支上新建他们自己的分支,他们可以随意提交代码到自己的分支,而其他人不会受到影响。只有在将来合并的时候再去处理有冲突的地方。
  如果是用SVN,每次你提交之前,都需要先update服务器上的代码,然后再commit,如果这时候有冲突,那你就不得不停下来先解决冲突的问题,在继续开发。实际上很有可能这个冲突可能已经分配给其他的同事去处理修改了。但是你并不知道。所以你现在不改也不行,改了有可能会影响到其他人的工作。(git允许你先把你要实现的都实现,最后大家都做好了,再来讨论代码的合并删减)。


新建,合并分支

在上一段的末尾,其实我们已经聊到了这个问题。在我们的项目中如果有了诸多分支之后,我们如何来对有冲突的分支进行合并呢?

首先我们创建一个新的项目,然后在这个项目master分支上新建一个iss53分支和hotfix分支用来修复不同的问题。

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit     (master)
$ git branch iss53#创建分支iss53
Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git branch hotfix#创建分支hotfix
Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git branch#显示当前所有分支
  hotfix
  iss53
* master

现在我们切换到hotfix分支上,假设我们需要解决hotfix这个问题就是需要在hotfix上新建一个文件。

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git checkout hotfix#切换到hotfix分支
D       README.md
A       README.txt
D       READMORE
Switched to branch 'hotfix'

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ ls#显示当前目录下的文件
gitignore  README.txt

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ vim hotfix.txt#创建文件hotfix.txt

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ ls#显示创建后的结果
gitignore  hotfix.txt  README.txt
        Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ git add hotfix.txt #将新建的文件加到暂存区
warning: LF will be replaced by CRLF in hotfix.txt.
The file will have its original line endings in your working directory.

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ git status#查看暂存区
On branch hotfix
Changes to be committed:
  (use "git reset HEAD <file>..." to unstage)

        renamed:    README.md -> README.txt
        deleted:    READMORE
        new file:   hotfix.txt


Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ git commit -m"add hotfix.txt"#提交暂存区文件
[hotfix b75aa14] add hotfix.txt
 3 files changed, 1 insertion(+), 12 deletions(-)
 rename README.md => README.txt (100%)
 delete mode 100644 READMORE
 create mode 100644 hotfix.txt

现在就可以看到我们已经成功创建hotfix.txt这个文件到branch Hotfix下,并且已经成功提交。按照我们规定的假设,现在hotfix这个问题已经解决了。那么我们来把hotfix分支和master分支合并到一起。

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (hotfix)
$ git checkout master      #先切换到master分支
Switched to branch 'master'
Your branch is up-to-date with 'origin/master'.

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git merge hotfix       #合并master和hotfix分支
Updating 7bd6f76..b75aa14
Fast-forward
 README.md => README.txt |  0
 READMORE                | 12 ------------
 hotfix.txt              |  1 +
 3 files changed, 1 insertion(+), 12 deletions(-)
 rename README.md => README.txt (100%)
 delete mode 100644 READMORE
 create mode 100644 hotfix.txt

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git branch -d hotfix#删除多余的hotfix分支
Deleted branch hotfix (was b75aa14).

觉得这里有必要贴上官方文档的一句话:

请注意,合并时出现了“Fast forward”的提示。由于当前 master分支所在的提交对象是要并入的 hotfix 分支的直接上游,Git 只需把 master 分支指针直接右移。换句话说,如果顺着一个分支走下去可以到达另一个分支的话,那么 Git 在合并两者时,只会简单地把指针右移,因为这种单线的历史分支不存在任何需要解决的分歧,所以这种合并过程可以称为快进(Fast forward)。

现在我们整个项目的状态应该是下面这张图所示:


我们暂时回到另一个分支“iss53”。这个分支跟之前我们已经删掉的hotfix是一个妈生的。然后我们也对它进行一些操作。比如新建一个iss53.txt文件。

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git checkout iss53 #切换到分支iss53
Switched to branch 'iss53'

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (iss53)
$ vim iss53.txt #创建文件iss53.txt

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (iss53)
$ ls  #查看文件目录
gitignore  iss53.txt  README.md  READMORE

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (iss53)
$ git add iss53.txt #添加到暂存区
warning: LF will be replaced by CRLF in iss53.txt.
The file will have its original line endings in your working directory.

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (iss53)
$ git commit -m"add new file iss53.txt"#提交
[iss53 866f987] add new file iss53.txt
 1 file changed, 1 insertion(+)
 create mode 100644 iss53.txt

这一步结束之后,我们看看项目结构:


这时候,我们可以看到,iss53跟master已经不是直接的父子关系了,他们有个共同的父节点C2.也就是我们在创建hotfix分支和iss53分支之前的master。然后我们要对这两个分支进行合并。

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (iss53)
$ git checkout master#切换到master分支
Switched to branch 'master'
Your branch is ahead of 'origin/master' by 1 commit.
  (use "git push" to publish your local commits)

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ git merge iss53#合并分支iss53
error: Empty commit message.
Not committing merge; use 'git commit' to complete the merge.

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master|MERGING)
$ git commit -m"we try to merge the iss53 to master"#提交&添加提交说明
[master a723ad4] we try to merge the iss53 to master

Dendoink@QELRM4A8Y4JCC8E MINGW64 /e/GITPUSH/moretestaboutgit (master)
$ ls#查看结果
gitignore  hotfix.txt  iss53.txt  README.txt

直接引用原文的话:
   这次,Git 没有简单地把分支指针右移,而是对三方合并后的结果重新做一个新的快照,并自动创建一个指向它的提交对象(C6)(见图 3-17)。这个提交对象比较特殊,它有两个祖先(C4 和 C5)。
  值得一提的是 Git 可以自己裁决哪个共同祖先才是最佳合并基础;这和 CVS 或 Subversion(1.5 以后的版本)不同,它们需要开发者手工指定合并基础。所以此特性让 Git 的合并操作比其他系统都要简单不少。

结果如图:

如果合并时遇到问题呢?

原文档中会有例子,但是我一直尝试重现没有成功,每次我把新的branch里面的某个文件(master中也有这个文件)修改之后,Git都会在我的进行merge的时候直接用新的文件替换旧的。这姑且算是有个没弄明白的地方吧。这里不对官方文档中的处理方式进行描述。

由于篇幅的问题,我把关于Branch的其他内容放到了下一篇日志里面。先到这里。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,176评论 5 469
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,190评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,232评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,953评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,879评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,177评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,626评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,295评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,436评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,365评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,414评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,096评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,685评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,771评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,987评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,438评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,032评论 2 341

推荐阅读更多精彩内容

  • Git是目前最流行的版本管理系统,也是最先进的分布式版本控制系统(distributed version cont...
    pro648阅读 5,664评论 1 17
  • 纯手工打造每一篇开源资讯与技术干货,数十万程序员和Linuxer已经关注 1 Git 分支 - 分支简介 有人把 ...
    尘世不扰阅读 687评论 0 3
  • 随着移动互联网的深度普及,很多行业都在发生巨大的变化,新媒体与传统媒体形成对比补充,知识网红与娱乐明星形成类同差异...
    慧眼识鱼阅读 1,044评论 2 5
  • 大家好,我是日记星球229号星宝宝新疆玉,我正在参加日记星球第五期21天的蜕变之旅,这是我的第8篇原创日记! 参加...
    琢玉台阅读 315评论 0 0
  • 不知什么原因,这节语文课心情很是糟糕,感觉寂寞无聊透顶。要知道,我是很钟情于文学的,自然不可能抵触语文,但不安的心...
    百古风流阅读 243评论 0 0