1 - 本地版本库
创建版本库
版本库又名仓库,英文名repository,可以简单理解成一个目录,这个目录里面的所有文件都可以被Git管理起来,每个文件的修改、删除,Git都能跟踪,以便任何时刻都可以追踪历史,或者在将来某个时刻可以“还原”。
创建一个版本库
第一步,创建一个空目录:
1 | $ mkdir learngit |
pwd
命令用于显示当前目录。目前这个仓库位于/c/Users/86178/learngit
第二步,通过git init
命令把这个目录变成Git可以管理的仓库:
1 | $ git init |
瞬间Git就把仓库建好了,而且告诉你是一个空的仓库(empty Git repository),当前目录下多了一个.git
的目录,这个目录是Git来跟踪管理版本库的。如果没有看到.git
目录,那是因为这个目录默认是隐藏的,用ls -ah
命令就可以看见。
把文件添加到版本库
所有的版本控制系统,其实只能跟踪文本文件的改动,比如TXT文件,网页,所有的程序代码等等。 而图片、视频这些二进制文件,虽然也能由版本控制系统管理,但没法跟踪文件的变化。
注意: 不要使用Windows自带的记事本编辑任何文本文件。原因是Microsoft开发记事本的团队在保存UTF-8编码的文件时,在每个文件开头添加了0xefbbbf(十六进制)的字符,因此会遇到很多不可思议的问题。
现在我们编写一个readme.txt
文件, 放到learngit
目录下(子目录也行),因为这是一个Git仓库,放到其他地方Git无法找到这个文件。
第一步,用命令git add
告诉Git,把文件添加到仓库:
1 | $ git add readme.txt |
执行上面的命令,没有任何显示,说明添加成功。
注:
1 | git add -A 提交所有变化 |
第二步,用命令git commit
告诉Git,把文件提交到仓库:
1 | $ git commit -m "wrote a readme file" |
git commit
命令:-m
后面输入的是本次提交的说明,这样你就能从历史记录里方便地找到改动记录。
git commit
命令执行成功后会告诉你,1 file changed
:1个文件被改动(我们新添加的readme.txt文件);2 insertions
:插入了两行内容(readme.txt有两行内容)。
commit
可以一次提交很多文件,所以可以多次add
不同的文件,比如:
1 | $ git add file1.txt |
git commit -am && -m
我们需要用git add命令来跟踪新文件,但如果使用git commit -am可以省略使用git add命令将已跟踪文件放到暂存区的功能
git commit -m用于提交暂存区的文件,git commit -am用于提交跟踪过的文件
添加修改
修改readme.txt文件,将第一行的Git is a version control system.
改为Git is a distributed version control system.
git status
命令可以让我们时刻掌握仓库当前的状态。运行git status
命令看看结果:
1 | $ git status |
上面的命令输出告诉我们,readme.txt
被修改过了,但还没有准备提交的修改。
用git diff
这个命令可以看看具体修改了什么内容。
- 要明白这3个概念,工作区(working tree),暂存区(index /stage),本地仓库(repository)
- git跟不同的参数,比较不同的区间的版本。
- git diff:是查看working tree与index的差别的。
- git diff –cached:是查看index与repository的差别的。
- git diff HEAD:是查看working tree和repository的差别的。其中:HEAD代表的是最近的一次commit的信息。
1 | $ git diff readme.txt |
git diff
顾名思义就是查看difference,显示的格式正是Unix通用的diff格式,可以从上面的命令输出看到,我们在第一行添加了一个distributed
单词。
提交修改和提交新文件是一样的两步,第一步是git add
:
1 | $ git add readme.txt |
同样没有任何输出。在执行第二步git commit
之前,我们再运行git status
看看当前仓库的状态:
1 | $ git status |
git status
告诉我们,将要被提交的修改包括readme.txt
,下一步,就可以放心地提交了:
1 | $ git commit -m "add distributed" |
提交后,我们再用git status
命令看看仓库的当前状态:
1 | $ git status |
Git告诉我们当前没有需要提交的修改,而且,工作目录是干净(working tree clean)的。
版本回退
HEAD
指向的版本就是当前版本,因此,Git允许我们在版本的历史之间穿梭,使用命令git reset --hard commit_id
。- 穿梭前,用
git log
可以查看提交历史,以便确定要回退到哪个版本。 - 要重返未来,用
git reflog
查看命令历史,以便确定要回到未来的哪个版本。
git log
命令查看历史记录:
1 | $ git log |
git log
命令显示从最近到最远的提交日志,我们可以看到3次提交,最近的一次是append GPL
,上一次是add distributed
,最早的一次是wrote a readme file
。
如果嫌输出信息太多,可以加上--pretty=oneline
参数:
1 | $ git log --pretty=oneline |
一大串类似1094adb...
的是commit id
(版本号),和SVN不一样,Git的commit id
不是1,2,3……递增的数字,而是一个SHA1计算出来的一个非常大的数字,用十六进制表示,而且不同人的commit id
肯定不一样。因为Git是分布式的版本控制系统,后面我们还要研究多人在同一个版本库里工作,如果大家都用1,2,3……作为版本号,那肯定就冲突了。
版本回退
在Git中,用HEAD
表示当前版本,也就是最新的提交,上一个版本就是HEAD^
,上上一个版本就是HEAD^^
,当然往上100个版本写100个^
比较容易数不过来,所以写成HEAD~100
。
现在,我们要把当前版本append GPL
回退到上一个版本add distributed
,就可以使用git reset
命令:
1 | $ git reset --hard HEAD^ |
cat查看readme.txt
的内容:
1 | $ cat readme.txt |
当你用$ git reset --hard HEAD^
回退到add distributed
版本时,再想恢复到append GPL
,就必须找到append GPL
的commit id。Git提供了一个命令git reflog
用来记录你的每一次命令:
1 | $ git reflog |
这样,就可以重返未来
1 | $ git reset --hard 1094a |
工作区和暂存区
工作区(Working Directory)
就是在电脑里能看到的目录,比如learngit
文件夹就是一个工作区:
版本库(Repository)
工作区有一个隐藏目录.git
,这个不算工作区,而是Git的版本库。
Git的版本库里存了很多东西,其中最重要的就是称为stage(或者叫index)的暂存区,还有Git为我们自动创建的第一个分支master
,以及指向master
的一个指针叫HEAD
。
需要提交的文件修改用git add
把文件添加进暂存区,然后, 用git commit
是把暂存区的所有修改提交到当前分支。因为我们创建Git版本库时,Git自动为我们创建了唯一一个master
分支,所以,现在,git commit
就是往master
分支上提交更改。
一旦提交后,如果你又没有对工作区做任何修改,那么工作区就是“干净”的:
修改
管理修改
Git跟踪并管理的是修改,而非文件。 每次修改,如果不用git add
到暂存区,那就不会加入到commit
中。
撤销修改
场景1:当你改乱了工作区某个文件的内容,想直接丢弃工作区的修改时,用命令git checkout -- file
。
命令git checkout -- readme.txt
意思就是,把readme.txt
文件在工作区的修改全部撤销,这里有两种情况:
一种是
readme.txt
自修改后还没有被放到暂存区,现在,撤销修改就回到和版本库一模一样的状态;一种是
readme.txt
已经添加到暂存区后,又作了修改,现在,撤销修改就回到添加到暂存区后的状态。
总之,就是让这个文件回到最近一次git commit
或git add
时的状态。
场景2:当你不但改乱了工作区某个文件的内容,还添加到了暂存区时,想丢弃修改,分两步,第一步用命令git reset HEAD file
,就回到了场景1,第二步按场景1操作。
场景3:已经提交了不合适的修改到版本库时,想要撤销本次提交,参考<版本回退>一节,不过前提是没有推送到远程库。
删除文件
删除工作区文件:可以在文件管理器中直接删除,或使用rm
命令删除:$ rm test.txt
- 一、确实要从版本库删除该文件
就用命令git rm
删掉,并且git commit
:
1 | $ git rm test.txt |
现在,文件就从版本库中被删除了。
注意:删除文件夹是
1 | git rm -r images/ |
- 二、误删
可以把误删的文件恢复到最新版本:
1 | $ git checkout -- test.txt |
git checkout
其实是用版本库里的版本替换工作区的版本,无论工作区是修改还是删除,都可以“一键还原”。
注意:从来没有被添加到版本库就被删除的文件,是无法恢复的!