zoukankan      html  css  js  c++  java
  • git暂缓区工作区原理和修改删除等命令

    前言:
     
    在前面几篇,特别是第(3)篇已经详细的说了几个很重要也很常见的命令了。这一节接着说。同样也是参考了这本书:
     
    http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000
     
    工作区和暂存区的关系及区别
     
    前面我们多次提到命令git status命令,他的一些状态变化,其实都与现在我们将要说的工作区和暂存区有关,再说这个之前呢,我们先来理清楚几个名词概念,会更好理解工作区和暂存区。
     
    工作区
     
    工作区(working Directory),就是我们项目的目录,也就是我们例子中的learngit 目录,这就是我们所说的工作区,比较简单。如下图所示:
     
     
    这个目录就是一个工作区,我们需要版本的控制的文件都放到这个文件夹下面。
     
    版本库
     
    版本库(Repository) ,工作区有一个隐藏目录“.git”,如上图所示,这个目录不算工作区,而是Git的版本库。我们打开这个.git文件夹,里面有很多文件:
     

     
    其中最重要的就是称为stage(或者叫index)的暂存区,还有Git为我们自动创建的第一个分支master,以及指向master的一个指针叫HEAD。
     
    暂缓区
     
    我们先来看一张图,用来区分工作区和暂缓区以及分支的概念:
     

     
    上图中,工作区就是我们的本地目录,也就是learngit目录,stage就是暂缓区,master分支就是主分支。
     
    我们先把这三个关系说一下,然后我们再具体的例子来说明:
     
    前面讲了我们把文件往Git版本库里添加的时候,是分两步执行的:

    第一步是用“git add”把文件添加进去,实际上就是把文件修改添加到暂存区

    第二步是用“git commit”提交更改,实际上就是把暂存区的所有内容提交到当前分支

    因为我们创建Git版本库时,Git自动为我们创建了唯一一个master分支,所以,现在,commit就是往master分支上提交更改。

    你可以简单理解为,需要提交的文件修改通通放到暂存区,然后,一次性提交暂存区的所有修改。
     
    写个例子
     
    好,现在我们来写个例子来说明下工作区,暂存区以及分支之间的关系。我们主要是用git status这个命令,来查看。
     
    我们先来修改一个README.md文件,加一行内容:Git has a mutable index called stage.
    1.  
      Git is a distributed version control system.
    2.  
      Git is free software. it is a good tool. distributed under the GPL.
    3.  
      Git has a mutable index called stage.
    然后,在工作区,也就是learngit 目录下,新增一个LICENSE文本文件(内容随便写)。我们拷贝一段apache的:
    Apache License
                               Version 2.0, January 2004
                            http://www.apache.org/licenses/
    
    
       TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
    
    好。这个时候,我们使用一下git status这个命令,来查看下:
    $ git status
    On branch master
    Changes not staged for commit:
      (use "git add <file>..." to update what will be committed)
      (use "git checkout -- <file>..." to discard changes in working directory)
    
    
            modified:   README.md
    
    
    Untracked files:
      (use "git add <file>..." to include in what will be committed)
    
    
            LICENSE
    
    
    no changes added to commit (use "git add" and/or "git commit -a")
    
    我们注意观察这2段内容,它的表述是不一样的。README.md是说 Changes not staged for commit,意思有改动还没修改加到暂存区,刚好我们前面提到暂存区就是stage,所以not staged就是没有到暂存区。LICENSE是说Untracked files ,意思是还没添加追踪,因为我们是新加的,从没git add,所以没被git版本库管理起来。
     
    好,那我们就分2次用git add命令将这2次加到暂存区,也就stage中。
    tonyyang@021ZJ1315 /d/learngit (master)
    $ git add README.md
    tonyyang@021ZJ1315 /d/learngit (master)
    $ git add LICENSE
    
    ok,之前说过没有任何输出代表成功了。现在就是将工作区(working Directory)的代码提交到了暂存区(stage),所以这一步的操作,我们画图就是这样:
     

     
    ok,添加到暂存区之后,我们再来git status这个命令查看下,状态是啥样。
    tonyyang@021ZJ1315 /d/learngit (master)
    $ git status
    On branch master
    Changes to be committed:
      (use "git reset HEAD <file>..." to unstage)
    
    
            new file:   LICENSE
            modified:   README.md
    
    它提示我们2个文件,一个是new新的,一个modified修改过,都已经提交到暂缓区,但是都没提交commit到master分支。也就是说现在暂存区是有东西的。
     
    那我们就提交到分支吧,还记得怎么提交分支嘛?对,用这个命令:git commit -m "some msg"
    $ git commit -m "modify README.md and add a new LICENSE file"
    [master b4a73ed] modify README.md and add a new LICENSE file
     2 files changed, 6 insertions(+)
     create mode 100644 LICENSE
    
    ok,commit成功了。提示我们成功了。那么,现在的暂存区状态是什么样子呢?我们也来画图看看:
     

     
    我们通过上诉图,很清楚的看到了。暂存区现在是空的了,工作区也已经是干净的了,代码已经提交到当前的master分支上去了。是不是这样呢。同样,我们用git status这个命令查看下
    $ git status
    On branch master
    nothing to commit, working directory clean
    
    它提示我们都已经提交了,工作区也是干净的。
     
     
    git跟踪的是修改
     
    上面很详细的说了暂存区的一些基本的东西,下面我们通过管理修改的例子,说学习一下为毛git这么快,比其他的版本工具来优秀,因为git跟踪并管理的是修改,并不是文件。
     
    git 中如何定义修改呢:你新增了一行,这就是一个修改,删除了一行,也是一个修改,更改了某些字符,也是一个修改,删了一些又加了一些,也是一个修改,甚至创建一个新文件,也算一个修改。
     
    SVN中是跟踪的是这个文件,只要这个文件发生变化,我们就认为是有diff的。但是GIT跟踪的是修改,一个修改只要没被add到缓存区,都不算diff。
     
    我们直接看例子吧:
     
    第一步,对README.md做一个修改,比如加一行内容:  Git tracks changes.

    $ vi README.md
    Git is a distributed version control system.
    Git is free software. it is a good tool. distributed under the GPL.
    Git has a mutable index called stage.
    Git tracks changes.
    
    第二步:我们用git add 命令将这次修改添加到暂存区。并用git status这个命令查看下状态
     
    $ git add README.md
    $ git status
    On branch master
    Changes to be committed:
      (use "git reset HEAD <file>..." to unstage)
    
    
            modified:   README.md
    
    提示我们已经添加到了暂存区。
     
    第三步:我们再继续修改下这个文件:
     
    $ vi README.md
    Git is a distributed version control system.
    Git is free software. it is a good tool. distributed under the GPL.
    Git has a mutable index called stage.
    Git tracks changes of files.
    
    第四步:我们不对第三步的修改添加到暂存区,我们这一步直接提交到master分支,用git commit -m "msg"
     
    $ git commit -m "git tracks changes"
    [master a2f0cbc] git tracks changes
     1 file changed, 1 insertion(+)
    
    ok,提示我们已经commit成功,我们现在再git status这个命令查看下状态
     
    $ git status
    On branch master
    Changes not staged for commit:
      (use "git add <file>..." to update what will be committed)
      (use "git checkout -- <file>..." to discard changes in working directory)
    
    
            modified:   README.md
    
    
    no changes added to commit (use "git add" and/or "git commit -a")
    我们发现,第三步我们的修改 Git tracks changes of files. 还存放在工作区,没有被commit。为毛会这样呢?别激动,我们一开始不就说了嘛:git 跟踪管理的是你的每一次修改,并不是这个文件。
     
    如果你还不清楚,我们理一下刚才这个例子的顺序:
     
    第一次修改 -> git add -> 第二次修改(没有git add) -> git commit

    你看,我们前面讲了,Git管理的是修改,当你用“git add”命令后,在工作区的第一次修改被放入暂存区,准备提交,但是,在工作区的第二次修改并没有放入暂存区,所以,“git commit”只负责把暂存区的修改提交了,也就是第一次的修改被提交了,第二次的修改不会被提交。
     
    由于,第二次修改还在工作区,根据前面讲的,我们是可以用git diff这个命令来看差异的。
     
    $ git diff
    diff --git a/README.md b/README.md
    index f4febdd..0262b58 100644
    --- a/README.md
    +++ b/README.md
    @@ -1,4 +1,4 @@
     Git is a distributed version control system.
     Git is free software. it is a good tool. distributed under the GPL.
     Git has a mutable index called stage.
    -Git tracks changes.
    +Git tracks changes of files.
    
    可见,第二次修改确实没有被提交。
     

    那怎么提交第二次修改呢?你可以继续add再commit,也可以别着急提交第一次就commit修改,先add第二次修改,再commit,就相当于把两次修改合并后一块提交了:

    第一次修改 -> add -> 第二次修改 -> add -> commit

    如何撤销修改
     
    通过上面的N个步骤学习,我们渐渐的掌握了git中的基本命令,也清楚了git的工作方式,git中的工作区暂存区的联系,那么,我们现在来看看如何撤销的修改。
     
    这个场景很常见,一般我们使用版本管理也是觉得这个功能很赞。注意,这里的撤销与之前咱们讲到的版本管理还是有点区别的,版本回退是已经提交commit到master分支了。但是我们现在说的这个撤销有点类似于SVN 你的 reset 命令,还没有commit 到分支。
     
    由于是还没提交到master 分支,所以就分为2种情况:1. 还在工作区,还没add到暂存区,2. 已经add到暂存区了。
     
    我们来看下第一种情况:还在工作区。
     
    还在工作区
     
    比如我看小王不爽了,很郁闷就在README.md里骂了一句,xiaowang is sb.
    Git is a distributed version control system.
    Git is free software. it is a good tool. distributed under the GPL.
    Git has a mutable index called stage.
    Git tracks changes of files.
    xiaowang is sb.
    
    现在想想不应该骂他的,反悔了。那我如何撤销我刚才的修改呢?你说我一行行的手动去掉行不行?当然可以。但是这样做就真的是sb了。要是工作量太大,那不还把人搞死啊。
     
    我们用可以先git status这个命令查看下状态
    $ git status
    On branch master
    Changes not staged for commit:
      (use "git add <file>..." to update what will be committed)
      (use "git checkout -- <file>..." to discard changes in working directory)
    
    
            modified:   README.md
    
    
    no changes added to commit (use "git add" and/or "git commit -a")
    
    正是说明了我刚才加的还在工作区,还没add到暂存区。它也提示我们 use "git checkout -- <file>..." to discard changes in working directory
     
    正是这个命令:git checkout -- file来撤销一个文件的修改。注意一定要加 -- 。不然就是创建分支了。后面我们会讲到。
     
    $ git checkout -- readme.txt
    我们试一下:
     
    $ git checkout -- README.md
    $ git status
    On branch master
    nothing to commit, working directory clean
    
    $ vi README.md
    Git is a distributed version control system.
    Git is free software. it is a good tool. distributed under the GPL.
    Git has a mutable index called stage.
    Git tracks changes.
    
    OK,恢复到修改之前了。
     
    已经add加到暂存区
     
    第二种情况,就是手贱了,已经加到暂存去了。咋搞呢。也好搞
     
    还是刚才这个例子,已经add 到暂存区了:
     
    $ git status
    On branch master
    Changes to be committed:
      (use "git reset HEAD <file>..." to unstage)
    
    
            modified:   README.md
    
    提示我们已经加到暂存区了,准备提交。
     
    我们如果到了这一步,我们也不用慌,我们肯定有办法的,你看,方法不是已经告诉你了嘛,上面仔细看:use git reset HEAD <file>.. to unstage
     
    命令git reset HEAD file命令,可以把暂存区的修改撤销掉(unstage),重新放回工作区。这个命令同样可用来回退版本,之前第三篇已经讲过了:git reset --hard commit_id(id前7位)或者 git reset --hard HEAD^来回退版本。当我们用HEAD时,表示最新的版本。
     
    好,我们来试一下:
     
    $ git reset HEAD README.md
    Unstaged changes after reset:
    M       README.md
    
    ok,提示我们撤销unstage成功。我们再看下status
     
    $ git status
    On branch master
    Changes not staged for commit:
      (use "git add <file>..." to update what will be committed)
      (use "git checkout -- <file>..." to discard changes in working directory)
    
    
            modified:   README.md
    
    
    no changes added to commit (use "git add" and/or "git commit -a")
    
    非常完美,已经撤销到工作区了。那你如果也想工作区里的也撤销咋搞。那就是前面的第一种情况了,一步步的来:
     
    $ git checkout -- README.md
    $ git status
    On branch master
    nothing to commit, working directory clean
    
    $ vi README.md
    Git is a distributed version control system.
    Git is free software. it is a good tool. distributed under the GPL.
    Git has a mutable index called stage.
    Git tracks changes.
    如果,你更手贱,已经commit到master分支了。擦得。这样就更悲剧了。但是方法也有的,就是前面讲的回退了。我还是说一下吧,免得你记不住:
     
    $ git reflog
    29e9c55 HEAD@{1}: commit: xiaowang is sb
    a2f0cbc HEAD@{2}: commit: git tracks changes
    b4a73ed HEAD@{3}: commit: modify README.md and add a new LICENSE file
    c790395 HEAD@{4}: reset: moving to c790395
    0e87e34 HEAD@{5}: reset: moving to HEAD^
    c790395 HEAD@{6}: reset: moving to c790395
    0e87e34 HEAD@{7}: reset: moving to HEAD^
    c790395 HEAD@{8}: commit: add GPL
    0e87e34 HEAD@{9}: commit: it is a good tool
    4b86146 HEAD@{10}: commit: modify a line
    c96a8e5 HEAD@{11}: commit (initial): write a new readme.md file
    
    $ git reset --hard a2f0cbc
    HEAD is now at a2f0cbc git tracks changes
    
    ok。恢复了。
     
    要是你用push 命令推送了远程服务器,抱歉,各种记录已经推送到远程了。小王要知道了,估计就马上打屎你了。阿门!
     
     
    删除文件
     
    删除文件同样是很常见的场景,我们可以将一个文件在版本库你删除,同样如果是不小心删除了,也可以恢复。
     
    比如,我们新建一个文件test.txt文件,来用于演示我们删除要求。
     
    
    
    $ vi test.txt
    this file is will delete
    
    然后我们将其添加到版本库:
     
    $ git add test.txt
    $ git commit -m "test.txt"
    [master 4b8661a] test.txt
     1 file changed, 1 insertion(+)
     create mode 100644 test.txt
    
    然后,我们手动将其删除:
     
    $ rm test.txt
    $ git status
    On branch master
    Changes not staged for commit:
      (use "git add/rm <file>..." to update what will be committed)
      (use "git checkout -- <file>..." to discard changes in working directory)
    
    
            deleted:    test.txt
    
    
    no changes added to commit (use "git add" and/or "git commit -a")
    
    我们已经捕捉到text.txt被删除了。如果我们是确认删除,我们可以用git rm file这个命令删掉,并且git commit -m "msg"提交确认。如果你没有用git rm 命令,直接commit会提示,而且会提交不成功。
     
    $ git rm test.txt
    rm 'test.txt'
    $ git commit -m "delete test"
    [master f6a084e] delete test
     1 file changed, 1 deletion(-)
     delete mode 100644 test.txt
    
    好。ok就删除掉了。
     
    如果你后悔了不想删除,一种是还没commit,用上面知道的修改撤销就可以搞定。
    $ git checkout -- test.txt
    
    一种是已经commit到分支了的,就只能借助于版本回退了:
    $ git reset --hard HEAD^
    好了。搞定。
     
    本节总结
     
    这一节,主要是对git的暂存区和工作区,进行了剖析,很详细,总结下:
     
    1. 暂存区是Git非常重要的概念,弄明白了暂存区,就弄明白了Git的很多操作到底干了什么。
    2. git 管理的是修改,不是文件,所以commit只会提交已经add的暂缓区的修改
    3. 修改:当你改乱了工作区某个文件的内容,想直接丢弃工作区的修改时,用命令git checkout -- file

    4 .修改:当你不但改乱了工作区某个文件的内容,还添加到了暂存区时,想丢弃修改,分两步,第一步用命令git reset HEAD file,在用git checkout -- file

    5. 修改:已经提交了不合适的修改到版本库时,想要撤销本次提交,要用 git reset -- hard commit_id 来版本回退

    6. git rm用于你确认删除一个文件

     
  • 相关阅读:
    Sqlserver 2008:sp_msforeachdb 坑爹的错误陷阱
    安装官方 Synaptics 驱动,终于解决 HP Pavilion G4 笔记本 讨厌的触摸板锁定问题!
    Sqlserver 2008+:变更数据捕获(CDC) 和 更改跟踪(CT)
    PHP.ini 性能优化
    PHP Notice: Undefined index: ... 问题的解决方案
    查询类方法代码分析
    页面跳转方法总结大全
    如何关闭searchIndexer.exe进程
    如何正确运用PHP ini_set函数
    PHP中的串行化变量和序列化对象
  • 原文地址:https://www.cnblogs.com/zst062102/p/12772162.html
Copyright © 2011-2022 走看看