git使用手册

一、电脑本地初始化一个仓库

1. git init: 初始化一个电脑上本地仓库

终端进入项目目录,输入:

$ git init  

该命令将创建一个名为 .git 的子目录,这个子目录含有你初始化的 Git 仓库中所有的必须文件,这些文件是 Git 仓库的骨干。
详细介绍:https://git-scm.com/book/zh/v2/Git-基础-获取-Git-仓库

2. 创建远程仓库: 存放代码

目前我知道的git网站仓库:

码云:
https://gitee.com/
公有仓库、私有仓库都免费使用, 国内访问速度快。

github:
https://github.com/
公有仓库免费使用, 私有仓库收费, 有时候访问速度慢。

Bitbucket:
https://bitbucket.org/
公有仓库、私有仓库都免费使用, 但是免费的最多只能有5个用户对仓库进行读写,超过的就需要付费。
访问速度有时很慢。

gitlab:
公司自建的git服务器,随意使用。

创建仓库都差不多,在网站中点击新建仓库,然后选择仓库的类型(公有、私有),然后点击创建即可。

3. 本地Git添加远程仓库地址

git remote add <shortname> <url>: 添加仓库

本地git初始化后,此时还没有添加远程仓库地址,需要添加一个远程仓库地址才能上传代码到服务器。
可在终端中运行git remote add <shortname> <url> 添加一个新的远程 Git 仓库。
shortnameurl的简写,当上传代码的时候,可用这个简写代替url地址。
命令详细介绍:https://git-scm.com/book/zh/v2/Git-基础-远程仓库的使用

$ git remote add pb https://github.com/paulboone/ticgit

git remote -v:查看本地git的远程仓库地址

当添加好远程仓库后,可以使用命令来查看添加的仓库是否正确。

# 列出你指定的每一个远程服务器的简写
$ git remote
origin  

# 指定选项 -v,会显示需要读写远程仓库使用的 Git 保存的简写与其对应的 URL。
$ git remote -v
origin  https://github.com/schacon/ticgit (fetch)
origin  https://github.com/schacon/ticgit (push)

git remote show [remote-name] :查看远程仓库详细信息

如果想要查看某一个远程仓库的更多信息,可以使用 git remote show [remote-name] 命令。
如果想以一个特定的缩写名运行这个命令,例如 origin,会得到像下面类似的信息:

$ git remote show origin
* remote origin
  Fetch URL: https://github.com/schacon/ticgit
  Push  URL: https://github.com/schacon/ticgit
  HEAD branch: master
  Remote branches:
    master                               tracked
    dev-branch                           tracked
  Local branch configured for 'git pull':
    master merges with remote master
  Local ref configured for 'git push':
    master pushes to master (up to date)

远程仓库的重命名、移除、修改地址

如果想要重命名引用的名字可以运行 git remote rename 去修改一个远程仓库的简写名。
例如,想要将 pb 重命名为 paul,可以用 git remote rename 这样做:

$ git remote rename pb paul
$ git remote
origin
paul

这同样也会修改你的远程分支名字。 那些过去引用 pb/master 的现在会引用 paul/master。

如果因为一些原因想要移除一个远程仓库 - 可以使用 git remote rm :

$ git remote rm paul
$ git remote
origin

修改地址:

git remote set-url origin www.baidu.com

关于远程仓库详细介绍

二、从远程仓库(服务器)获取代码:本地什么也没有,拉取代码

1. 获取默认分之代码:git clone [URL]

如果本地没有代码,远程仓库有代码,则需要从远程仓库克隆代码。
克隆仓库的命令格式是 git clone [url] 。 比如,要克隆 Git 的可链接库 libgit2,可以用下面的命令:

$ git clone https://github.com/libgit2/libgit2   

这会在当前目录下创建一个名为 “libgit2” 的目录,并在这个目录下初始化一个 .git 文件夹,
从远程仓库拉取下所有数据放入 .git 文件夹,然后从中读取最新版本的文件的拷贝。
如果你进入到这个新建的 libgit2 文件夹,你会发现所有的项目文件已经在里面了,
准备就绪等待后续的开发和使用。

① 拉取仓库时自定义本地仓库文件名: git clone [URL] [新名字]

如果你想在克隆远程仓库的时候,自定义本地仓库的名字,你可以使用如下命令:

$ git clone https://github.com/libgit2/libgit2 mylibgit    

这将执行与上一个命令相同的操作,不过在本地创建的仓库名字变为 mylibgit。

关于克隆远程仓库命令详细介绍:https://git-scm.com/book/zh/v2/Git-基础-获取-Git-仓库

2. 直接拉取特定分之的代码:git clone -b [分支名] [代码地址]

如果不想克隆默认分之的代码, 也可以克隆特定分之的代码:

$ git  clone http://192.168.102.9/jas-paas/cloudlink-front-framework.git
Cloning into 'cloudlink-front-framework'...
remote: Counting objects: 14436, done.
remote: Compressing objects: 100% (3072/3072), done.
remote: Total 14436 (delta 11224), reused 14226 (delta 11075)
Receiving objects: 100% (14436/14436), 17.93 MiB | 4.98 MiB/s, done.
Resolving deltas: 100% (11224/11224), done.

3. 直接拉取特定标签(tag)的代码:git clone -b [标签名] [代码地址]

git允许直接克隆特定分之的代码,不过克隆好后,项目里没有分之,需要自己创建一个分支:

$ git clone -b V2.2 http://192.168.102.9/jas-paas/cloudlink-front-framework.git
Cloning into 'cloudlink-front-framework'...
remote: Counting objects: 14436, done.
remote: Compressing objects: 100% (3072/3072), done.
remote: Total 14436 (delta 11224), reused 14226 (delta 11075)
Receiving objects: 100% (14436/14436), 17.93 MiB | 4.70 MiB/s, done.
Resolving deltas: 100% (11224/11224), done.
Note: checking out 'eeea534cdae1f82c48c7b0de8f9993b54ffa065d'.

You are in 'detached HEAD' state. You can look around, make experimental
changes and commit them, and you can discard any commits you make in this
state without impacting any branches by performing another checkout.

If you want to create a new branch to retain commits you create, you may
do so (now or later) by using -b with the checkout command again. Example:

  git checkout -b <new-branch-name>

从信息中就可以看见,克隆特定标签的项目后,是没有分之的,需要进入项目目录后,使用命令创建一个分支:

$ git checkout -b V2.2

4. 如果已经拉取了代码,还需要拉取其他分支代码:git checkout -b [分支名] [远程仓库名]/[分支名]

112289165-WX20170420-162329.png

例如,本地就一个master分支,远程有2个分支(master,develop),把远程的develop拉取到本地:

# 本地的分支是干净的,也就是没有修改的文件
# 获取远程所有分支名字
~ git fetch
# 显示远程所有分支名字
~ git branch -a
# 提取远程新分支到本地
~ git checkout -b develop origin/develop       

三、记录每次更新到仓库

1. GIT中文件状态介绍(已跟踪/未跟踪)

工作目录下每个文件只有两种状态:

已跟踪的两种状态

指那些被纳入了版本控制的文件,在上一次快照中有它们的记录,在工作一段时间后,它们的状态可能处于未修改,已修改或已放入暂存区。
初次克隆某个仓库的时候,工作目录中的所有文件都属于已跟踪文件,并处于未修改状态。
git会自动管理已跟踪的文件,记录文件处于什么状态中。

已暂存(Changes to be committed)

文件在这个状态下的,说明已经准备好把文件提交了,可以把这次代码变动记录保存在历史记录中。
这个状态为GIT可以提交的内容。

已修改(Changes not staged for commit)

这状态下的文件, git只是知道修改了那些内容,但是并不会在提交代码的时候把这部分内容提交上去。
如果需要提交这部分代码需要使用命令git add 把文件添加到 已暂存中,然后提交代码。

未跟踪(Untracked files)

工作目录中除已跟踪文件以外的所有其它文件都属于未跟踪文件,它们既不存在于上次快照的记录中,也没有放入暂存区。
git不会去管理这些文件。

2. git status: 查看当前文件状态

使用git status命令可以查看文件处于什么状态,例如:

$ git status
On branch P02                                   // 告诉你当前是哪个分之下
Your branch is up-to-date with 'origin/P02'.    // 当前分之是从哪个仓库更新的
Changes to be committed:                        // 已暂存状态
  (use "git reset HEAD <file>..." to unstage)   // 使用这个命令可回退到Changes not staged for commit:

    modified:   fileName.ts

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:   test.txt

Untracked files:                                // 未跟踪的文件
  (use "git add <file>..." to include in what will be committed)

    src/library/

git status命令输出的信息中:

Changes to be committed:
已暂存状态
如果此时提交,那么该文件此时此刻的版本将被留存在历史记录中。

Changes not staged for commit:
已跟踪文件的内容发生了变化,但还没有放到暂存区。

Untracked files:
未跟踪的文件,意味着 Git 在之前的快照(提交)中没有这些文件;
Git 不会自动将之纳入跟踪范围,除非你明明白白地告诉它“我需要跟踪该文件”,
这样的处理让你不必担心将生成的二进制文件或其它不想被跟踪的文件包含进来。

3. git status -s紧凑格式输出

使用git status -s命令或git status --short命令,你将得到一种更为紧凑的格式输出。

$ git status -s           
 M README              // 出现在右边的 M 表示该文件被修改了但是还没放入暂存区
MM Rakefile            // 文件已经放入暂存区,但是又修改过了,在已修改中也存在
A  lib/git.rb          // 新添加到暂存区中的文件前面有 A 标记
M  lib/simplegit.rb    // 出现在靠左边的 M 表示该文件被修改了并放入了暂存区
?? LICENSE.txt         // 新添加的未跟踪文件前面有 ?? 标记

4. GIT代码管理:改变文件的状态

git add

git add命令是个多功能命令:可以用它开始跟踪新文件,或者把已跟踪的文件放到暂存区,
还能用于合并时把有冲突的文件标记为已解决状态等。
将这个命令理解为“添加内容到下一次提交中”而不是“将一个文件添加到项目中”要更加合适。
例子:
src/test.txt文件是一个未跟踪文件,可使用git add命令开始跟踪:

$ git add src/test.txt    //指定单个文件添加
$ git add src/*           //指定src目录下所有文件都添加

git checkout -- <file>: 撤消对文件的修改

如果你并不想保留对 CONTRIBUTING.md 文件的修改,
将它还原成上次提交时的样子(或者刚克隆完的样子,或者刚把它放入工作目录时的样子)
可使用如下命令:

git checkout -- CONTRIBUTING.md

注:
你需要知道 git checkout -- [file] 是一个危险的命令,这很重要。
你对那个文件做的任何修改都会消失 - 你只是拷贝了另一个文件来覆盖它。
除非你确实清楚不想要那个文件了,否则不要使用这个命令。

git reset HEAD <file>:取消暂存的文件

如果你在提交代码的时候,不想提交一些文件,可使用该命令把文件从暂存中回退到已修改的文件中。

git reset HEAD CONTRIBUTING.md     

关于撤销操作的详细介绍: https://git-scm.com/book/zh/v2/Git-基础-撤消操作

5. .gitignore:忽略文件,不纳入GIT版本控制

一般我们总会有些文件无需纳入 Git 的管理,也不希望它们总出现在未跟踪文件列表。
通常都是些自动生成的文件,比如日志文件,或者编译过程中创建的临时文件等。
在这种情况下,我们可以在项目根目录创建一个名为 .gitignore 的文件,列出要忽略的文件模式。

文件 .gitignore 的格式规范如下:

  1. 所有空行或者以 # 开头的行都会被 Git 忽略。
  2. 可以使用标准的 glob 模式匹配。
  3. 匹配模式可以以(/)开头防止递归。
  4. 匹配模式可以以(/)结尾指定目录。
  5. 要忽略指定模式以外的文件或目录,可以在模式前加上惊叹号(!)取反。

一个 .gitignore 文件的例子:

# 忽略所有后缀为 .a 的文件
*.a
# 但是要跟踪lib.a,即使你上面忽略了 .a文件
!lib.a
# 仅忽略当前目录的TODO文件夹, 不是subdir/TODO 
/TODO
# 忽略build /目录中的所有文件
build/
# 忽略doc / notes.txt,而不是doc / server / arch.txt
doc/*.txt
# 忽略doc 目录中的所有.pdf文件
doc/**/*.pdf

GitHub 有一个十分详细的针对数十种项目及语言的 .gitignore 文件列表:
https://github.com/github/gitignore

这部分详细介绍: https://git-scm.com/book/zh/v2/Git-基础-记录每次更新到仓库

6. git diff: 查看代码修改详细内容

如果你想知道文件具体修改了什么,可以用git diff命令。

Ⅰ. git diff: 查看尚未暂存的文件修改记录

git diff命令查看修改之后还没有暂存起来的变化内容。也就是git status命令输出信息中,
Changes not staged for commit:下面的文件, 例如:

$ git diff         //查看所有详细修改
$ git diff 1.txt   //只查看 1.txt文件的 详细修改   

//输出信息如下   
diff --git a/.gitignore b/.gitignore   //①
index e722882..f98470c 100644          //②
--- a/.gitignore    //③
+++ b/.gitignore    //③
@@ -3,8 +3,8 @@     //④

 *~
 *.sw[mnpcod]
-.DS_Store          //⑤
 *.log              //⑥
+text.t             //⑦
 *.tmp
 *.tmp.*
 log.txt

① git格式的diff,进行比较的是,a版本的.gitignore(即变动前)和b版本的.gitignore(即变动后)。
② 表示两个版本的git哈希值(index区域的e722882对象,与工作目录区域的f98470c对象进行比较),
最后的六位数字是对象的模式(普通文件,644权限)。
③ 表示进行比较的两个文件。 "---"表示变动前的版本,"+++"表示变动后的版本。
④ 下面代码所在的行:
-3,7: 修改前下面信息变化的行, 从第3行开始一直到第7行
+3,6: 修改后下面信息变化的行, 从第3行开始一直到第6行
⑤ 以- 开头表示删除的代码
⑥ 以空格开头表示没有修改
⑦ 以+开头便是增加的代码

git diff输出结果介绍

Ⅱ. git diff --staged: 查看以暂存文件修改记录

查看已暂存的将要添加到下次提交里的内容可是使用命令git diff --staged(GIT版本1.6.1以上)。
也可以使用git diff --cached命令。

git diff --staged        // 查看所有已暂存文件的修改记录
git diff --staged t.txt  // 只查看1.txt文件 已暂存的修改记录

Ⅲ. git diff 的其他用法: --stat HEAD SHA1

~ git diff --stat         // 查看简单的diff结果,只查看修改的文件名、修改了多少内容
~ git diff HEAD           // 查看所有修改记录(已暂存、已修改):显示工作版本(Working tree)和HEAD的差别
~ git diff topic master   // 直接将两个分支上最新的提交做diff
~ git diff HEAD^ HEAD     // 比较上次提交commit和上上次提交
~ git diff SHA1 SHA2      // 比较两个历史版本之间的差异

7. 把代码纳入版本控制中: 提交更新 git commit

当所有需要提交的代码都放到暂存区后,就可以提交代码了:

$ git commit

这种方式会启动文本编辑器以便输入本次提交的说明。
一般都是 vim 或 emacs。使用 git config --global core.editor 命令设定你喜欢的编辑软件。
编辑器会显示类似下面的文本信息(本例选用 Vim 的屏显方式展示):

# Please enter the commit message for your changes. Lines starting
# with '#' will be ignored, and an empty message aborts the commit.
# On branch master
# Changes to be committed:
#   new file:   README
#   modified:   CONTRIBUTING.md
#
~
~
".git/COMMIT_EDITMSG" 9L, 283C

可以在这段文本信息中看见, 所有以#符号开头的行,都会被忽略,也就是#是注释行,不会放到提交信息中去。
接下来就可以输入一些提交信息了,关于写提交信息的建议:

信息应当以少于 50 个字符(25个汉字)的单行开始且简要地描述变更,接着是一个空白行,再接着是一个更详细的解释。
Git 项目要求一个更详细的解释,包括做改动的动机和它的实现与之前行为的对比 - 这是一个值得遵循的好规则。

提交信息简单的模板:

修改的摘要(50 个字符或更少)

如果必要的话,加入更详细的解释文字。在
大概 72 个字符的时候换行。在某些情形下,
第一行被当作一封电子邮件的标题,剩下的
文本作为正文。分隔摘要与正文的空行是
必须的(除非你完全省略正文);如果你将
两者混在一起,那么类似变基等工具无法
正常工作。

关于VIM编辑器使用教程: vim编辑器常用命令与用法总结

commit命令的其他的用法: -m选项,提交信息简写

在 commit 命令后添加 -m 选项,将提交信息与命令放在同一行,如下所示:

$ git ci -m "测试提交"
[master f949399] 测试提交
 1 file changed, 1 insertion(+)

从输出的信息中可以看见:
当前是在哪个分支(master)提交的,本次提交的完整 SHA-1 校验和是什么(f949399),
以及在本次提交中,有多少文件修订过,多少行添加和删改过。

注意:
提交时记录的是放在暂存区域的快照。 任何还未暂存的仍然保持已修改状态,可以在下次提交时纳入版本管理。
每一次运行提交操作,都是对你项目作一次快照,以后可以回到这个状态,或者进行比较。

commit命令的其他的用法: -a选项,提交所有修改过的文件(已暂存,已修改)

有时候把修改的提交到暂存区,然后在提交代码比较繁琐, GIT提供了一种-a选项,在提交代码的时候,
跳过暂存区,把所有已经跟踪的文件暂存起来一并提交,从而跳过git add步骤,例如:

$ 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:   CONTRIBUTING.md

no changes added to commit (use "git add" and/or "git commit -a")

$ git commit -a -m 'added new benchmarks'
[master 83e38c7] added new benchmarks
 1 file changed, 5 insertions(+), 0 deletions(-)

从输出信息中可以看到:提交之前不再需要 git add 文件“CONTRIBUTING.md”了。

关于git commit命令详解:https://git-scm.com/book/zh/v2/Git-基础-记录每次更新到仓库

8. git push: 把本地的跟新推送到远程仓库服务器上

当在本地开发完毕,需要把代码提交到服务器时,可使用it push [remote-name] [branch-name]命令。
你想要将 master 分支推送到 origin 服务器时:

$ git push origin master     // 把更新上传到 origin服务器的 master分之上。
$ git push                   // 上传本地所有分支代码到远程对应的分支上

只有当你有所克隆服务器的写入权限,并且之前没有人推送过时,这条命令才能生效。
当你和其他人在同一时间克隆,他们先推送到上游然后你再推送到上游,你的推送就会毫无疑问地被拒绝。
你必须先将他们的工作拉取下来并将其合并进你的工作后才能推送。

四、 查看提交历史: git log

在提交了若干更新,又或者克隆了某个项目之后,你也许想回顾下提交历史。 完成这个任务最简单而又有效的工具是 git log 命令:

$ git log
commit ca82a6dff817ec66f44342007202690a93763949    // 提交的 SHA-1 校验和
Author: Scott Chacon <schacon@gee-mail.com>        // 作者的名字和电子邮件地址
Date:   Mon Mar 17 21:52:11 2008 -0700             // 提交时间

    changed the version number                     // 提交说明

commit 085bb3bcb608e1e8451d4b2432f8ecbe6306e7e7
Author: Scott Chacon <schacon@gee-mail.com>
Date:   Sat Mar 15 16:40:33 2008 -0700

    removed unnecessary test

默认不用任何参数的话,git log 会按提交时间列出所有的更新,最近的更新排在最上面。

git log 的常用选项: -stat, -p, --name-status, --pretty=oneline, --abbrev-commit

git log --stat:显示每次更新的文件修改统计信息(修改的文件名,每个文件添加的多少、删除了多少数字)。

git log --name-only: 只显示修改的文件名,没有其他信息

git log --pretty=oneline: 用一行显示信息

$ git log 1.txt            //查看 1.txt文件的历史修改记录。
$ git log -n               //只查看前n次修改记录,例如查看前2次的记录: git log -2
$ git log -p               // 按补丁格式显示每个更新之间的差异。  
$ git log --shortstat      // 只显示 --stat 中最后的行数修改添加移除统计。
$ git log --name-status    // 显示新增、修改、删除的文件清单。     
$ git log --abbrev-commit  // 仅显示 SHA-1 的前几个字符,而非所有的 40 个字符。
$ git log --relative-date  // 使用较短的相对时间显示(比如,“2 weeks ago”)。
$ git log --graph          // 显示 分支合并历史。   
$ git log --pretty=oneline // 用一行显示信息
// 使用其他格式显示历史提交信息。可用的选项包括 oneline,short,full,fuller 和 format(后跟指定格式)。

关于git log命令详解

五、 打标签(一个版本发布) git tag

像其他版本控制系统(VCS)一样,Git 可以给历史中的某一个提交打上标签,以示重要。
比较有代表性的是人们会使用这个功能来标记发布版本节点(v1.0 等等)。

1. 什么是标签?

标签可以认为是版本管理工具的一个版本,也就是说像很多第三方开源软件安装包表示的版本号,
比如说1.0,2.0等等,它代表修复了一个bug,或者重大的重构,git的标签也是这样的一个作用,
可以对版本的不同阶段做一个标示。

2. 查看现有的标签

① 查看所有标签:git tag

$ git tag   
v0.1
v1.3
v1.4

这个命令以字母顺序列出标签;但是它们出现的顺序并不重要。

② 查看符合条件的标签:git tag -l 'v1.*'

也可以使用特定的模式查找标签, 如果只对 1.0以上tag感兴趣,可以运行:

$ git tag -l 'v1.*'

③ 查看一个标签的详细信息:git tag show V1.4

该命令会输出打标签者的信息、打标签的日期时间、附注信息,然后显示具体的提交信息。

3. 创建标签:Git 使用两种主要类型的标签:

git tag -a:创建附注标签(annotated)

附注标签是存储在 Git 数据库中的一个完整对象。
它们是可以被校验的;其中包含打标签者的名字、电子邮件地址、日期时间;还有一个标签信息;
并且可以使用 GNU Privacy Guard (GPG)签名与验证。 通常建议创建附注标签,这样你可以拥有以上所有信息。

创建一个附注标签,使用tag命令时指定-a选项:

$ git tag -a v1.4 -m 'my version 1.4'    

-m选项指定了一条将会存储在标签中的信息。 如果没有为附注标签指定一条信息,Git 会运行编辑器要求你输入信息。

如果你有自己的私钥,还可以用 GPG 来签署标签,只需要把之前的-a改为-s(取 signed 的首字母)即可:

git tag -s v1.5 -m 'my signed 1.5 tag'

再运行 git show 会看到对应的 GPG 签名也附在其内.

git tag V1.4: 创建轻量标签(lightweight)

如果你只是想用一个临时的标签,或者因为某些原因不想要保存那些信息,则可以选择使用轻量标签。
轻量标签本质上是将提交校验和存储到一个文件中 - 没有保存任何其他信息。
创建轻量标签,不需要使用 -a、-s 或 -m 选项,只需要提供标签名字:

$ git tag v1.4-lw

git tag -a v1.2 9fceb02:对某个历史提交打标签(后期打标签)

如果你在项目版本发布的时候忙忘记打标签,你可以在之后补上标签。
要在那个提交上打标签,你需要在命令的末尾指定提交的校验和(或部分校验和):

git tag -a v1.2 9fceb02    

4. 把创建的标签推送到服务器上:git push origin [tagname]

默认情况下,git push命令并不会传送标签到远程仓库服务器上。 在创建完标签后你必须显式地推送标签到共享服务器上。

$ git push origin v1.5

Counting objects: 14, done.
Delta compression using up to 8 threads.
Compressing objects: 100% (12/12), done.
Writing objects: 100% (14/14), 2.05 KiB | 0 bytes/s, done.
Total 14 (delta 3), reused 0 (delta 0)
To git@github.com:schacon/simplegit.git
 * [new tag]         v1.5 -> v1.5

如果想要一次性推送很多标签,也可以使用带有--tags选项的git push命令。
这将会把所有不在远程仓库服务器上的标签全部传送到服务器。

$ git push origin --tags
Counting objects: 1, done.
Writing objects: 100% (1/1), 160 bytes | 0 bytes/s, done.
Total 1 (delta 0), reused 0 (delta 0)
To git@github.com:schacon/simplegit.git
 * [new tag]         v1.4 -> v1.4
 * [new tag]         v1.4-lw -> v1.4-lw

现在,当其他人从仓库中克隆或拉取,他们也能得到你的那些标签。

5. 检出标签: 代码变成标签的样子

在 Git 中你并不能真的检出一个标签,因为它们并不能像分支一样来回移动。
如果你想要工作目录与仓库中特定的标签版本完全一样,
可以使用git checkout -b [branchname] [tagname]在特定的标签上创建一个新分支:

$ git checkout -b version2 v2.0.0
Switched to a new branch 'version2'

6. 删除标签

git tag -d v1.0:删除本地标签

$ git tag -d v1.0  
Deleted tag 'v1.0' (was 2ffea56)

git push origin :refs/tags/v0.1:删除服务器标签

$ git push origin :refs/tags/v0.1
To http://192.168.95.95/user/test.git
 - [deleted]         v0.1

如果服务器的标签删除后,本地的也需要自己删除。

关于git tag详解

六、 GIT分支

1. 分支介绍作用

Git 的分支,其实本质上仅仅是指向提交对象的可变指针。 Git 的默认分支名字是 master。
在多次提交操作之后,你其实已经有一个指向最后那个提交对象的 master 分支。 它会在每次的提交操作中自动向前移动。

分支作用:

  1. 开发多个项目任务,比如说我有两个任务都比较紧急,任务1需要两天完成,任务2需要一天完成,而任务1是之前就已经开始进行的,任务二是中间加的新任务,所以需要第一天就完成任务2.
  2. master分支始终要保证可发布的状态,用dev分支和bug分支进行开发和错误调试,这样能够保证主干代码的干净、可发布。
  3. 自己开发测试或者修复BUG等等,可以避免代码的丢失。

2. 本地分支的创建、切换、删除:

① 创建分支:git branch testing

比如,创建一个 testing 分支, 你需要使用 git branch 命令:

$ git branch testing

这条命令会根据当前的分支来创建一个新分支,分支中内容与当前分支内容完全一样。
git branch命令仅仅 创建 一个新分支,并不会自动切换到新分支中去。

② 切换分支:git checkout testing

要切换到一个已存在的分支,你需要使用git checkout命令。 我们现在切换到新创建的 testing 分支去:

$ git checkout testing 

③ 创建并且换:git checkout -b testing

$ git checkout -b testing
Switched to a new branch "testing"

这个命令会创建一个分支并切换到新分支中去, 实际上他是上面2条命令的简化版。

④ 删除分支:git branch -d testing

删除一个分支的时候,你不能在要删除的分支中,要切换到别的分支中,否则会报错。
如果要删除 testing分支,可以使用带 -d 选项的 git branch 命令来删除分支:

$ git br -d testing
Deleted branch testing (was 4baf2a3).

3. 远程仓库分支的新建与删除

①远程仓库分支的新建: git push origin testing

如果想把本地的新分支推送到远程仓库,可使用 git push [远程仓库名] [分支名]

$ git br -a

  testing
* master
  remotes/origin/master

$ git push origin testing

Total 0 (delta 0), reused 0 (delta 0)
To http://192.168.132.55/user/test.git
 * [new branch]      testing -> testing

② 删除远程仓库的无用分支:git push origin --delete testing

如果远程仓库某个分支无用了,想要删除,可以运行带有 --delete 选项的 git push 命令来删除一个远程分支:

git push origin --delete testing
To http://192.168.132.55/user/test.git
 - [deleted]         testing

4. 分支合并:git merge

如果一个分支的功能开发完毕了,需要把开发的内容合并的其他分支中去,则需要使用 git的 merge命令。
例如, 现在有2个分支(master, ningning), master分支的代码要合并到ningning,2个分支代码都已经git commit过了。
如果没有commit,需要先commit,否则不能合并代码。

① 合并命令:git merge [分支名]

// 切换到需要添加新功能的分支上
$ git checkout ningning
// 把 master分支上的代码合并到ningning分支上
$ git merge master

② 合并很完美,直接合并成功的提示

如果合并成功了则输出的信息类似于下面:

$ git merge master 
Updating f42c576..3a0874c
Fast-forward
 index.html | 2 ++
 1 file changed, 2 insertions(+)   

// 或者下面的输出信息
Merge made by the 'recursive' strategy.
index.html |    1 +
1 file changed, 1 insertion(+)

Git 将合并的结果做了一个新的快照并且自动创建一个新的提交指向它。

③ 合并过程中代码有冲突,解决冲突

3353411608-WX20170420-172823.png

错误信息类似于:

Auto-merging index.html
CONFLICT (content): Merge conflict in index.html
Automatic merge failed; fix conflicts and then commit the result.

Auto-merging是指已经自动合并的文件,
CONFLICT (content)是指有冲突的文件,需要解决冲突的文件。

打开有冲突的文件,冲突部分代码类似于下面:

<<<<<<< HEAD:ningning
<div id="footer">contact : email.support@github.com</div>
=======
<div id="footer">
 please contact us at support@github.com
</div>
>>>>>>> master  

<<<<< ======= 中的是 ningning分支的代码。
======= >>>>>> 中的是 master分支的代码。
经过对比后删除冲突部分的代码, 并把 <<< ==== >>>> 所在行全部删除。
保存后,使用git add添加修改的文件。
使用git commit命令来完成合并提交:

解决冲突文件:git checkout --ours/--theirs放弃其中一个分支的冲突代码

如果想放弃一个分支文件的冲突代码,只保留一个分支的代码,可使用如下命令。

假如:冲突文件名为 1.txt,要放弃master分支的修改,可使用如下命令:

$ git checkout --ours 1.txt

放弃ningning分支冲突代码,可使用如下命令:

$ git checkout --theirs 1.txt 

④ 取消合并

冲突太多,解决冲突乱了, 可取消合并:git reset --hard HEAD

如果冲突代码太多了,解决冲突代码过程中产生了混乱,想要重新合并,可使用下面命令取消这次合并:

git reset --hard HEAD
HEAD is now at 9e791f3 提交信息
合并已经commit,还没上传远程仓库,取消合并:git reset --hard HEAD^

如果合并的代码产生了错误,或者合并有问题,但是已经commit了,但是还没有把合并提交到远程仓库,则可以使用如下命令取消这次合并:

$ git reset --hard HEAD^ 



GIT的高级用法(平时用的很少,了解就好)




GIT工具: 储藏与清理

1. git stash : 修改的代码暂存起来,把工作目录变干净,方便切换分支等做一些其他事情

当你的工作已经做了一段时间后,所开发的功能还没有完成, 而这时候你想要切换分支做一点别的事情。
但你又不想应为过一会就回到这一点而为做了一半工作就创建一次提交,这个时候你就可以使用git stash命令。
git stash命令会处理工作目录的脏的状态,即修改的跟踪文件与暂存的改动。然后将未完成的修改保存到一个栈上,
而你可以在任何时候重新应用这些改动。

① 把代码储藏起来: git stash

$ git status
Changes to be committed:
  (use "git reset HEAD <file>..." to unstage)

    modified:   index.html

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:   lib/simplegit.rb

//储藏代码
$ git stash
Saved working directory and index state \
  "WIP on master: 049d078 added the index file"
HEAD is now at 049d078 added the index file
(To restore them type "git stash apply")

$ git status
# On branch master
nothing to commit, working directory clean

上面使用了git stash 储藏了代码,这时候在看工作目录就变成干净了,这个时候你就可以做别的事情了。
还可以使用git stash save命令,与 git stash 是一样的。

② 查看储藏的代码: git stash list

要查看储藏的东西,可以使用git stash list命令:

$ git stash list
stash@{0}: WIP on master: 049d078 added the index file
stash@{1}: WIP on master: c264051 Revert "added file_size"
stash@{2}: WIP on master: 21d80a5 added number to log

最新的是stash@{0}

③ 把储藏的代码取出来: git stash apply

如果想把储藏的代码应用回来,可使用git stash apply命令,如果不指定一个储藏,Git 认为指定的是最近的储藏。
如果想要应用其中一个更旧的储藏,可以通过名字指定它,像这样:git stash apply stash@{2}

$ git stash apply --index
# On branch master
# Changed but not updated:
#   (use "git add <file>..." to update what will be committed)
#
#      modified:   index.html
#      modified:   lib/simplegit.rb
#
git stash apply --index

文件的改动被重新应用了,但是之前已修改未暂存的文件却没有回到原来的位置,想要完全回到之前的样子,
必须使用--index选项来运行git stash apply命令,来尝试重新应用暂存的修改。
如果已经那样做了,那么你将回到原来的位置:

$ git stash apply --index
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#      modified:   index.html
#
# Changed but not updated:
#   (use "git add <file>..." to update what will be committed)
#
#      modified:   lib/simplegit.rb
#

④ 删除储藏堆栈上的某个储藏:git stash drop stash@{0}

使用git stash apply 命令只会尝试应用暂存的工作 ,在堆栈上还有它。
可以运行git stash drop加上将要移除的储藏的名字来移除它:

$ git stash list
stash@{0}: WIP on master: 049d078 added the index file
stash@{1}: WIP on master: c264051 Revert "added file_size"
stash@{2}: WIP on master: 21d80a5 added number to log
$ git stash drop stash@{0}
Dropped stash@{0} (364e91f3f268f0900bc3ee613f9f733e82aaed43)

⑤ 应用储藏然后立即从栈上扔掉它:git stash pop

这个命令只会应用最新的,然后丢掉它。

git stash --keep-index: 不要储藏任何你通过 git add 命令已暂存的东西

当你做了几个改动并只想提交其中的一部分,过一会儿再回来处理剩余改动时,这个功能会很有用。

$ git status -s
M  index.html
 M lib/simplegit.rb

$ git stash --keep-index
Saved working directory and index state WIP on master: 1b65b17 added the index file
HEAD is now at 1b65b17 added the index file

$ git status -s
M  index.html

git stash -u: 把任何创建的未跟踪文件也储藏

默认情况下,git stash只会储藏已经在索引中的文件。
如果指定 --include-untracked-u标记,Git 也会储藏任何创建的未跟踪文件。

$ git status -s
M  index.html
 M lib/simplegit.rb
?? new-file.txt

$ git stash -u
Saved working directory and index state WIP on master: 1b65b17 added the index file
HEAD is now at 1b65b17 added the index file

$ git status -s
$

从储藏创建一个分支:解决应用储藏冲突问题

如果储藏了一些工作,将它留在那儿了一会儿,然后继续在储藏的分支上工作,在重新应用工作时可能会有问题。
如果应用尝试修改刚刚修改的文件,你会得到一个合并冲突并不得不解决它。
如果想要一个轻松的方式来再次测试储藏的改动,可以运行git stash branch创建一个新分支,
检出储藏工作时所在的提交,重新在那应用工作,然后在应用成功后扔掉储藏:

$ git stash branch testchanges
Switched to a new branch "testchanges"
# On branch testchanges
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#      modified:   index.html
#
# Changed but not updated:
#   (use "git add <file>..." to update what will be committed)
#
#      modified:   lib/simplegit.rb
#
Dropped refs/stash@{0} (f0dfc4d5dc332d1cee34a634182e168c4efc3359)

清理工作目录

有时候你想清除工作目录一些无用的文件,或是为了运行一个干净的构建而移除之前构建的残留。
这个时候你可以使用git clean命令。 它被设计为从工作目录中移除未被追踪的文件。
如果你改变主意了,你也不一定能找回来那些文件的内容。
一个更安全的选项是运行 git stash --all 来移除每一样东西并存放在栈中。

git clean -f -d:移除工作目录中所有未追踪的文件以及空的子目录

-d 是 目录的意思。
-f 意味着 强制 或 “确定移除”。

$ git clean -f -d
Removing 5.txt
Removing tttt/

git clean -f -d -n: 做一次演习然后告诉你 将要 移除什么

git clean -f -d -x: 移除.gitiignore文件中忽略的文件

默认情况下,git clean 命令只会移除没有忽略的未跟踪文件。
任何与 .gitiignore 或其他忽略文件中的模式匹配的文件都不会被移除。
如果你也想要移除那些文件,例如为了做一次完全干净的构建而移除所有由构建生成的 .o 文件,
可以给 clean 命令增加一个 -x 选项。

$ git status -s
 M lib/simplegit.rb
?? build.TMP
?? tmp/

$ git clean -n -d
Would remove build.TMP
Would remove tmp/

$ git clean -n -d -x
Would remove build.TMP
Would remove test.o
Would remove tmp/

重写历史:git 提交历史修改

1. 修改最后一次提交

修改你最近一次提交可能是所有修改历史提交的操作中最常见的一个。
对于你的最近一次提交,你往往想做两件事情:修改提交信息,或者修改你添加、修改和移除的文件的快照。

① 只是修改最后一次提交的提交信息:git commit --amend

如果只是想修改最后一次提交的提交信息,可以直接使用如下命令:

$ git commit --amend 
[dev 3fa2e1a] 合并提交测试3号,测试修改最后一次提交。在测试一下。
 Date: Thu Nov 2 11:29:00 2017 +0800
 1 file changed, 1 insertion(+)
 create mode 100644 5.txt

这会把你带入文本编辑器,里面包含了你最近一条提交信息,供你修改。
当保存并关闭编辑器后,编辑器将会用你输入的内容替换最近一条提交信息。

② 修改最后一次提交的快照:git addgit rm然后git commit --amend

如果你已经完成提交,又因为之前提交时忘记添加一个新创建的文件,想通过添加或修改文件来更改提交的快照,
也可以通过类似的操作来完成。 通过修改文件然后运行git addgit rm一个已追踪的文件,
随后运行git commit --amend拿走当前的暂存区域并使其做为新提交的快照。

使用这个技巧的时候需要小心,因为修正会改变提交的 SHA-1 校验和。
它类似于一个小的变基 - 如果已经推送了最后一次提交则需要慎重修改。

③ 把修改的最后一次提交信息强推到服务器:git push --f

此操作要慎重,如果要修改服务器上的最后一次提交,可现在本地修改,
然后使用git push --f(force)强推到服务器中去。

git push --force
Counting objects: 3, done.
Delta compression using up to 8 threads.
Compressing objects: 100% (2/2), done.
Writing objects: 100% (3/3), 458 bytes | 458.00 KiB/s, done.
Total 3 (delta 0), reused 0 (delta 0)
To http://192.168.102.9/yulilong/test.git
 + 5c0a032...3fa2e1a dev -> dev (forced update)

④ gitlab服务器强推被拒绝: 仓库 -> 设置 -> 保护分支: 把被保护的分支去掉即可解决问题

2. 使用git rebase合并多次commit

使用git log 查看一下历史:

$ git log 

commit 3fa2e1a951ff823fbec625a96049e27ef35a85b8 (HEAD -> dev, origin/dev)
Author: yulilong <yulilong222q@outlook.com>
Date:   Thu Nov 2 11:29:00 2017 +0800

    合并提交测试3号,测试修改最后一次提交。在测试一下。

commit f526876def0b80676bfcf52937f78052e2d63955
Author: yulilong <yulilong222q@outlook.com>
Date:   Thu Nov 2 11:27:20 2017 +0800

    合并提交测试2号

commit 4d2b6ce4d5154c0b739e552e679bb7f4ce05fb2c
Author: yulilong <yulilong222q@outlook.com>
Date:   Thu Nov 2 11:25:59 2017 +0800

    合并提交测试一号

commit 9e791f38f4d5f472a371a8c147f37670185582f8
Author: yulilong <yulilong222q@outlook.com>
Date:   Mon Oct 30 10:22:02 2017 +0800

    这是dev分支上的合并提交测试

commit 4baf2a319c0de8f46630ba85db11cc4aebd1d2cd
Author: yulilong <yulilong222q@outlook.com>
Date:   Fri Sep 29 10:20:08 2017 +0800

    这是测试-a -m命令提交记录

如果想要合并最后三次的提交,可使用git rebase -i HEAD~3git rebase -i 9e791f38f4d命令来压缩.
该命令执行后,会弹出一个编辑窗口,3次提交的commit倒序排列,最上面的是最早的提交,最下面的是最近一次提交。

  1 pick 4d2b6ce 合并提交测试一号
  2 pick f526876 合并提交测试2号
  3 pick 3fa2e1a 合并提交测试3号,测试修改最后一次提交。在测试一下。
  4 
  5 # Rebase 9e791f3..3fa2e1a onto 9e791f3 (3 commands)
  6 #
  7 # Commands:
  8 # p, pick = use commit
  9 # r, reword = use commit, but edit the commit message
 10 # e, edit = use commit, but stop for amending
 11 # s, squash = use commit, but meld into previous commit
 12 # f, fixup = like "squash", but discard this commit's log message
 13 # x, exec = run command (the rest of the line) using shell
 14 # d, drop = remove commit
 15 #
 16 # These lines can be re-ordered; they are executed from top to bottom.
 17 #
 18 # If you remove a line here THAT COMMIT WILL BE LOST.
 19 #
 20 # However, if you remove everything, the rebase will be aborted.
 21 #
 22 # Note that empty commits are commented out

修改第二行、第三行的第一个单词pick为squash(这些命令什么意思下面的注释有说明)。
然后保存退出,git会压缩提交历史,
如果有冲突,需要修改,修改的时候要注意,保留最新的历史,不然我们的修改就丢弃了。
修改以后要记得敲下面的命令:

$ git add .  
# git rebase --continue  

如果你想放弃这次压缩的话,执行以下命令:

$ git rebase --abort 

如果没有冲突,或者冲突已经解决,则会出现如下的编辑窗口:

  1 # This is a combination of 3 commits.
  2 # This is the 1st commit message:
  3 
  4 合并提交测试一号
  5 
  6 # This is the commit message #2:
  7 
  8 合并提交测试2号
  9 
 10 # This is the commit message #3:
 11 
 12 合并提交测试3号,测试修改最后一次提交。在测试一下。
 13      
 14 # Please enter the commit message for your changes. Lines starting
 15 # with '#' will be ignored, and an empty message aborts the commit.
 16 #   
 17 # Date:      Thu Nov 2 11:25:59 2017 +0800
 18 #        
 19 # interactive rebase in progress; onto 9e791f3
 20 # Last commands done (3 commands done):
 21 #    squash f526876 合并提交测试2号
 22 #    squash 3fa2e1a 合并提交测试3号,测试修改最后一次提交。在测试一下。
 23 # No commands remaining.
 24 # You are currently rebasing branch 'dev' on '9e791f3'.
 25 #
 26 # Changes to be committed:
 27 #   modified:   1.txt
 28 #   new file:   4.txt
 29 #   new file:   5.txt

编辑好合并的提交信息后保存,可看见如下输出信息:

[detached HEAD 83d65c7] 三次合并提交的测试,成功了。
 Date: Thu Nov 2 11:25:59 2017 +0800
 3 files changed, 7 insertions(+)
 create mode 100644 4.txt
 create mode 100644 5.txt
Successfully rebased and updated refs/heads/dev.

最后把合并的记录强推到服务器中去:

git push -f
Counting objects: 5, done.
Delta compression using up to 8 threads.
Compressing objects: 100% (3/3), done.
Writing objects: 100% (5/5), 596 bytes | 596.00 KiB/s, done.
Total 5 (delta 1), reused 0 (delta 0)
To http://192.168.102.9/yulilong/test.git
 + 3fa2e1a...83d65c7 dev -> dev (forced update)

服务器合并提交前:

服务器合并后:

参考链接:http://blog.csdn.net/yangcs2009/article/details/47166361

3. 把一个文件从提交历史中彻底删除:git filter-branch --tree-filter 'rm -f 1.txt' HEAD

有些人不经思考使用git add .,意外地提交了一个巨大的二进制文件,你想将它从所有地方删除。
也许你不小心提交了一个包含密码的文件,而你想让你的项目开源。filter-branch大概会是你用来清理整个历史的工具。
如果想要从整个历史总删除1.txt文件,你可以在filter-branch上使用--tree-filter选项:

$ git filter-branch -f --tree-filter 'rm -f 1.txt' HEAD 
Rewrite 83d65c7b098e0ec1f14d9a332187632b49d2ad9f (5/5) (0 seconds passed, remaining 0 predicted)    
Ref 'refs/heads/dev' was rewritten

如果执行命令的时候提示如下错误,删除 .git/refs/original/ 目录,或使用-f命令强制覆盖:

$ git filter-branch -f --tree-filter 'rm -f 1.txt' HEAD 
Cannot create a new backup.
A previous backup already exists in refs/original/
Force overwriting the backup with -f   

git filter-branch -f --tree-filter 'rm -f 1.txt' HEAD 
Rewrite 83d65c7b098e0ec1f14d9a332187632b49d2ad9f (5/5) (0 seconds passed, remaining 0 predicted)    
Ref 'refs/heads/dev' was rewritten

如果后悔删除了,可使用如下命令恢复:git reset --hard refs/original/refs/heads/[分支名]

$ git reset --hard refs/original/refs/heads/dev
HEAD is now at 83d65c7 三次合并提交的测试,成功了。

把删除后的项目提交到服务器:git push origin +[分支名], 注意:一旦运行此命令,删除的文件不能找回

$ git push origin +dev         

Counting objects: 14, done.
Delta compression using up to 8 threads.
Compressing objects: 100% (9/9), done.
Writing objects: 100% (14/14), 1.40 KiB | 1.40 MiB/s, done.
Total 14 (delta 1), reused 0 (delta 0)
To http://192.168.102.110/user/test.git
 + 83d65c7...4b4da80 dev -> dev (forced update)

注意,分支名字前面的+号一定不能忘记,否则会报如下错误:

To http://192.168.102.110/user/test.git
 ! [rejected]        dev -> dev (non-fast-forward)
error: failed to push some refs to 'http://192.168.102.110/user/test.git'
hint: Updates were rejected because the tip of your current branch is behind
hint: its remote counterpart. Integrate the remote changes (e.g.
hint: 'git pull ...') before pushing again.
hint: See the 'Note about fast-forwards' in 'git push --help' for details.

推送之前要把服务器的分支保护去掉,否则会报如下错误:

remote: GitLab: You are not allowed to force push code to a protected branch on this project.
To http://192.168.102.110/user/test.git
 ! [remote rejected] dev -> dev (pre-receive hook declined)
error: failed to push some refs to 'http://192.168.102.110/user/test.git'

删除一堆类似文件:git filter-branch --tree-filter "find * -type f -name '*~' -delete" HEAD

全局性地更换电子邮件地址

$ git filter-branch --commit-filter '
        if [ "$GIT_AUTHOR_EMAIL" = "schacon@localhost" ];
        then
                GIT_AUTHOR_NAME="Scott Chacon";
                GIT_AUTHOR_EMAIL="schacon@example.com";
                git commit-tree "$@";
        else
                git commit-tree "$@";
        fi' HEAD

参考链接: https://git-scm.com/book/zh/v2/Git-工具-重写历史

GIT的工作原理

Git 的思维框架(将其作为内容管理器)是管理三棵不同的树。“树” 在我们这里的实际意思是 “文件的集合”,而不是指特定的数据结构。

Git 作为一个系统,是以它的一般操作来管理并操纵这三棵树的:

----------------------------
树                       用途

HEAD                    上一次提交的快照,下一次提交的父节点

index                   预期的下一次提交的快照

Working Directory       沙盒(工作目录)
  • HEAD

HEAD 是当前分支引用的指针,它总是指向该分支上的最后一次提交。 这表示 HEAD 将是下一次提交的父结点。
通常,理解 HEAD 的最简方式,就是将它看做 你的上一次提交 的快照。

下例命令就显示了HEAD快照实际的目录列表,以及其中每个文件的SHA-1校验和:

$ git cat-file -p HEAD
tree cfda3bf379e4f8dba8717dee55aab78aef7f4daf
author Scott Chacon  1301511835 -0700
committer Scott Chacon  1301511835 -0700

initial commit

$ git ls-tree -r HEAD
100644 blob a906cb2a4a904a152...   README
100644 blob 8f94139338f9404f2...   Rakefile
040000 tree 99f1a6d12cb4b6f19...   lib

cat-filels-tree是底层命令,它们一般用于底层工作,在日常工作中并不使用。不过它们能帮助我们了解到底发生了什么。

  • index

index(索引)是你的预期的下一次提交。也会将这个概念引用为 Git 的 “暂存区域”,这就是当你运行git commit时Git看起来的样子。

Git 将上一次检出到工作目录中的所有文件填充到索引区,它们看起来就像最初被检出时的样子。
之后你会将其中一些文件替换为新版本,接着通过 git commit 将它们转换为树来用作新的提交。

$ git ls-files -s
100644 a906cb2a4a904a152e80877d4088654daad0c859 0   README
100644 8f94139338f9404f26296befa88755fc2598c289 0   Rakefile
100644 47c6340d6459e05787f644c2447d2595f5d3a54b 0   lib/simplegit.rb

ls-files是底层命令,它会显示出索引当前的样子。
确切来说,索引并非技术上的树结构,它其实是以扁平的清单实现的。不过对我们而言,把它当做树就够了。

  • Working Directory

Working Directory(工作目录),最后,你就有了自己的工作目录。
另外两棵树以一种高效但并不直观的方式,将它们的内容存储在 .git 文件夹中。
工作目录会将它们解包为实际的文件以便编辑。 你可以把工作目录当做 沙盒。
在你将修改提交到暂存区并记录到历史之前,可以随意更改。

工作流程

+-------------------------------------------------------------------+
|   +--------------+      +---------------+      +--------------+   |
|   |  Working     |      |     Index     |      |     HEAD     |   |
|   |  Directory   |      |               |      |              |   |
|   +-----+--------+      +-------+-------+      +-------+------+   |
|         |                       |                      |          |
|         |                       | Checkout the project |          |
|         |  <-------------------------------------------+          |
|         |                       |                      |          |
|         |    Stage Files        |                      |          |
|         +---------------------> |                      |          |
|         |                       |      Commit          |          |
|         |                       +--------------------> |          |
|         |                       |                      |          |
+-------------------------------------------------------------------+
  1. 假设我们进入到一个新目录,其中有一个文件。 我们称其为该文件的 v1 版本,将它标记为蓝色。
    现在运行git init,这会创建一个 Git 仓库,其中的 HEAD 引用指向未创建的分支(master 还不存在)。

  2. 此时,只有工作目录有内容。 现在我们想要提交这个文件,所以用git add来获取工作目录中的内容,并将其复制到索引(Index)中。

  3. 接着运行git commit,它首先会移除索引中的内容并将它保存为一个永久的快照,
    然后创建一个指向该快照的提交对象,最后更新 master 来指向本次提交。

  4. 此时如果我们运行git status,会发现没有任何改动,因为现在三棵树完全相同。
    现在我们想要对文件进行修改然后提交它。 我们将会经历同样的过程;首先在工作目录中修改文件。
    我们称其为该文件的 v2 版本,并将它标记为红色。

  5. 如果现在运行 git status,我们会看到文件显示在 “Changes not staged for commit,” 下面并被标记为红色,
    因为该条目在索引与工作目录之间存在不同。 接着我们运行git add来将它暂存到索引中。

  6. 此时,由于索引和 HEAD 不同,若运行git status的话就会看到 “Changes to be committed” 下的该文件变为绿色.
    也就是说,现在预期的下一次提交与上一次提交不同。 最后,我们运行git commit来完成提交。

  7. 现在运行git status会没有输出,因为三棵树又变得相同了。
    切换分支或克隆的过程也类似。 当检出一个分支时,它会修改 HEAD 指向新的分支引用,
    将 索引 填充为该次提交的快照,然后将 索引 的内容复制到 工作目录 中。

git工作详细介绍

git重置git reset,代码回退操作介绍

1. 只移动HEAD(相当于取消上一次提交):git reset --soft HEAD~git reset --soft 99ad0ec

reset做的第一件事是移动HEAD的指向。reset移动HEAD指向的分支。
reset --soft本质上是撤销了上一次git commit命令。 当你在运行git commit时,Git会创建一个新的提交,
并移动HEAD所指向的分支来使其指向该提交。当你将它resetHEAD~HEAD 的父结点)时,
其实就是把该分支移动回原来的位置,而不会改变索引和工作目录。
现在你可以更新索引并再次运行git commit来完成git commit --amend所要做的事情了。

2. 移动HEAD,更新index: git reset HEAD~git reset --mixed HEAD~

它依然会撤销一上次 提交,但还会 取消暂存 所有的东西。 于是,我们回滚到了所有git addgit commit的命令执行之前。

3. 移动HEAD,更新index,更新工作目录(working Directory): git reset --hard HEAD~

必须注意,--hard标记是reset命令唯一的危险用法,它也是Git会真正地销毁数据的仅有的几个操作之一。
其他任何形式的reset调用都可以轻松撤消,但是--hard选项不能,因为它强制覆盖了工作目录中的文件。
在这种特殊情况下,我们的Git数据库中的一个提交内还留有该文件的v3版本,我们可以通过reflog来找回它。
但是若该文件还未提交,Git仍会覆盖它从而导致无法恢复。

reset命令会以特定的顺序重写这三棵树,在你指定以下选项时停止:

  1. 移动 HEAD 分支的指向 (若指定了 --soft,则到此停止)

  2. 使索引看起来像 HEAD (若未指定 --hard,则到此停止)

  3. 使工作目录看起来像索引

4. 使用git reflog命令来查看所有已经提交过的commit

5. 通过路径来重置:git reset file.txt

假如我们运行git reset file.txt(这其实是git reset --mixed HEAD file.txt的简写形式,
因为你既没有指定一个提交的SHA-1或分支,也没有指定--soft--hard),它会:

  1. 移动HEAD分支的指向 (已跳过)

  2. 让索引看起来像HEAD(到此处停止)

所以它本质上只是将file.txtHEAD复制到索引中。
它还有取消暂存文件的实际效果。
我们可以不让Git从HEAD拉取数据,而是通过具体指定一个提交来拉取该文件的对应版本。
我们只需运行类似于git reset eb43bf file.txt的命令即可。

git checkout 介绍

切换分支:git checkout [分支名]

运行git checkout [branch]与运行git reset --hard [branch]非常相似,
它会更新所有三棵树使其看起来像 [branch],不过有两点重要的区别。

首先不同于reset --hardcheckout对工作目录是安全的,它会通过检查来确保不会将已更改的文件弄丢。
其实它还更聪明一些。它会在工作目录中先试着简单合并一下,这样所有还未修改过的文件都会被更新。
reset --hard则会不做检查就全面地替换所有东西。

第二个重要的区别是如何更新HEADreset会移动HEAD分支的指向,而checkout只会移动HEAD自身来指向另一个分支。

例如,假设我们有masterdevelop分支,它们分别指向不同的提交;我们现在在develop上(所以HEAD指向它)。
如果我们运行git reset master,那么develop自身现在会和master指向同一个提交。
而如果我们运行git checkout master的话,develop不会移动,HEAD自身会移动。 现在HEAD将会指向master

所以,虽然在这两种情况下我们都移动HEAD使其指向了提交 A,但做法是非常不同的。
reset 会移动HEAD分支的指向,而checkout则移动HEAD自身。

放弃index与working Directory的改动:git checkout [分支名] file.txt

该命令不会移动HEAD,只会把HEAD中的代码恢复到index中,同时把工作目录文件也恢复到HEAD中代码的样子。

值放弃修改working Directory 工作目录中的修改:git checkout file.txt放弃单个修改或使用git checkout .放弃所有修改

这个命令只会放弃工作目录中的修改,已经提交到index中的修改则不会改动。

reset\checkout命令详细介绍

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

推荐阅读更多精彩内容

  • 一、 Git 分支简介 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着你可以把你的工作从开发主线上分...
    常大鹏阅读 2,931评论 2 41
  • git作为时下最流行的代码管理工具,Git权威指南总结了十条喜欢Git的理由: 异地协同工作; 现场版本控制; 重...
    古斟布衣阅读 1,817评论 0 12
  • (根据我的亲身经历改编) 我姓末,单名一个云字。 我与世无争,人畜无害,目前只是一家星级酒店的男服务生而已。 这回...
    Myron马阅读 507评论 4 5
  • 全社会都在越来越多的倡导家庭教育的重要性,很多家长朋友们就开始困惑了,既然家庭教育这么重要,那如何开启科学的家庭教...
    智慧爱家长学院阅读 464评论 0 0
  • 前言 这本书和Objective-C高级编程-iOS和OS X多线程和内存管理实在是iOS开发人员必读书. 实在是...
    Jerry4me阅读 2,630评论 8 41