Git 中的回退操作:reset 和 revert
Git 中回退有 reset
和 revert
,这两个的区别就是是否保留更改记录
假设当前的提交情况是:A <- B <- C <- D <- HEAD
,如下图:
当前是 D
,希望回退到 A
,那我们可以使用 reset
命令,reset 后再看 git log 就会发现:B <- C <- D
宛如没有出现过,这适用于想完全舍弃 A
之后的修改
但是如果我们想保留 B <- C <- D
的修改记录,可能这三个 commit 的功能只是暂时用不到,以后可能还用到,或者可能当前分支是一个公共分支,B <- C <- D
可能已经被同步到了其他小伙伴电脑上,为了尽量避免代码冲突。这些情况就需要使用 revert
命令,这样会重新生成新的 commit,其中包含回退的记录(假设 D
这个 commit 是添加了一些代码,那么 revert D
的 commit 就是删除这些代码)
reset
使用 git reset A
,reset 会修改 head 的指向,这样可以回滚到 A
,默认使用的参数是 --mixed
,这个参数决定了 reset
时 Git 该如何处理工作区和暂存区
一般地,我们对于代码的修改会体现在 working tree
(工作区)的变动,通过 git add
添加即将要提交的文件到 index
(暂存区),通过 git commit
再提交到 repository
(本地仓库),如下图:
查看帮助:git help reset
git reset [<mode>] [<commit>]
This form resets the current branch head to <commit> and possibly updates the index (resetting it to the
tree of <commit>) and the working tree depending on <mode>. If <mode> is omitted, defaults to --mixed. The
<mode> must be one of the following:
--soft
Does not touch the index file or the working tree at all
--mixed
Resets the index but not the working tree
--hard
Resets the index and working tree
假设我们现在处在 D
,那么分别会有三种 reset
的情况
- 我们执行
git reset --soft A
,字面意思,轻柔地 reset,只将repository
回滚到了A
,而working tree
、index
维持reset
之前的状态,保持不变,这个时候直接执行commit
,这时候会得到一个和D
修改内容相同的 commitD'
(二者的 commit id 是不相同的),--soft
很适合呈现多次 commit 的修改的叠加效果,假设B
、C
、D
都是针对某一个功能的修改,其中的 commit 可能修改了同一个文件,想整合这些 commit 都修改了哪些内容,就可以使用--soft
从D
reset 到A
,那么B
、C
、D
的修改都会出现在index
中
- 我们执行
git reset --mixed A
,repository
、index
会回滚,working tree
维持reset
之前的状态,这个时候直接 commit,将无法提交,因为repository
与index
都被回滚了,二者是相同的,没有变化则无法提交 commit
- 我们执行
git reset --hard A
,按照参数hard
的字面意思,reset 的非常强硬,repository
、index
、working tree
都会回滚到A
,因为working tree
工作区也回滚了,所以本地的所有修改也将丢失,--hard
相对来说比较危险,需要确保工作区没有需要保留的代码,--hard
适合的情况是对于当前的即将要提交的代码失去信心,准备推倒重来
revert
如果想在回滚的同时保留 commit 记录,就需要使用 revert,revert 就是生成原 commit 逆向修改的 commit,从而实现会滚。当前是 D
,希望回退到 A
,就需要按顺序依次 revert D
、C
、B
:
git revert D
git revert C
git revert B
每一次 revert 都会生成新的 commit,需要依次手动输入 commit message,也可以先 revert 最后集中 commit
git revert --no-commit D
git revert --no-commit C
git revert --no-commit B
git commit -m " Revert D C B"
使用 revert 需要注意,如果即将 revert 的 commit 是一个 merge commit,那么会出现错误
使用 reset 方式,创建回滚的 commit
如果需要保留回滚记录,但是需要 revert 的 commit 是 merge commit,那就必须手动指定 mainline
,比较麻烦,可以用 reset 的方式创建回滚 commit,这种方式不受 merge 方式的影响:
git reset --hard A
git reset --soft D
git commit -m " Revert D C B"
通过这种方式也能回滚回 A
,并且生成一个新的 commit,其中包括了 D
、C
、B
的逆向修改:
先
reset --hard
到A
,这时repository
、index
、working tree
都会回滚到A
再
reset --soft
到D
,这时repository
指向了D
,但是index
和working tree
还保持在A
,当前即将 commit 的就是B
、C
、D
逆向修改的叠加
参考资料
git reset soft,hard,mixed之区别深解
What's the difference between git reset --mixed, --soft, and --hard?
How can I revert multiple Git commits?
相关阅读:Git 常见操作梳理
Git 中的回退操作:reset 和 revert的更多相关文章
- 大话git中的撤销操作
下面以现实场景作为情境. 基础知识,理解git中的几个区域 本地代码已经add,未commit 修改本地工作目录中的readme.md,添加文字"第一次修改" 然后查看下状态 ➜ ...
- 『现学现忘』Git基础 — 23、Git中的撤销操作
目录 1.撤销操作说明 2.撤销工作区中文件的修改 3.撤销暂存区中文件的修改 4.总结 1.撤销操作说明 我们在使用Git版本管理时,往往需要撤销某些操作.比如说我们想将某个修改后的文件撤销到上一个 ...
- git中的文件操作
现在我们的机器上有了一个 真实项目 的 Git 仓库,并从这个仓库中检出了所有文件的 工作副本. 通常,你会对这些文件做些修改,每当完成了一个阶段的目标,想要将记录下它时,就将它提交到仓库. git中 ...
- git中通过实际操作来了解常用命令
基本的6个命令 常用的就下面6个命令,但是详细的可能有上百个命令. 还需要特别了解git的几个名词,workspace:工作区,Index/Stage:暂存区,Respository:本地仓库,Rem ...
- Git 中的一些其他常用命令
1.查看提交的历史版本(git log) 我们可以使用 git log 命令来查看提交的历史版本. 默认不用任何参数的话,git log 会按提交时间列出所有的更新,最近的更新排在最上面.每个版本都有 ...
- Git命令之回退篇 git revert git reset
Git command之回退篇 欲练回退 必先了解:HEAD.index.WorkingCopy HEAD: 当前所在的分支版本顶端的别名,也就是最新的一次commit. git commit 之后与 ...
- git中reset与revert的使用
http://alpha-blog.wanglianghome.org/2010/07/30/git-partial-rollback/ reset(版本撤回) 格式 git reset [-q] [ ...
- Git回退---reset和revert
今天学习了git回退的两个命令,现在总结一下: 1.git reset 如果想回退错误的提交C和D,只要把指针移到B上 git reset --hard a0fvf8 而这时候,远程仓库的指针还在D上 ...
- git reset 版本回退操作
1 git回退命令 git reset --hard GIT_HEAD GIT_HEAD是你具体要回退的分支: 如图: 注: 查询GIT_HEAD可以通过两个命令:git log 获取未删除 ...
随机推荐
- Springmvc01-什么是Springmvc
首先,我们回顾一下什么是MVC 1.什么是MVC MVC是模型(model),视图(View),控制器(Controller)的简写,是一种软件基本规范 Model(模型):数据模型,提供要展示的 ...
- zookeeper篇-初识zookeeper
点赞再看,养成习惯,微信搜索「小大白日志」关注这个搬砖人. 文章不定期同步公众号,还有各种一线大厂面试原题.我的学习系列笔记. 什么是zookeeper? 中间件 作用于分布式系统 支持java.c语 ...
- asyncio 异步编程
首先了解一下协程,协程的本质就是一条线程,多个任务在一条线程上来回切换,协程的所有切换都是基于用户,只有在用户级别才能感知到的 IO 才会用协程模块来规避,在 python 中主要使用的协程模块是 a ...
- 一文说透 MySQL JSON 数据类型(收藏)
JSON 数据类型是 MySQL 5.7.8 开始支持的.在此之前,只能通过字符类型(CHAR,VARCHAR 或 TEXT )来保存 JSON 文档. 相对字符类型,原生的 JSON 类型具有以下优 ...
- docker 灵活的构建 php 环境
地址: https://github.com/ydtg1993/server 使用docker搭建灵活的线上php环境 有时候你可能不太需要一些别人已经集成了的包或者镜像 ...
- C语言函数调用栈
C语言函数调用栈 栈溢出(stack overflow)是最常见的二进制漏洞,在介绍栈溢出之前,我们首先需要了解函数调用栈. 函数调用栈是一块连续的用来保存函数运行状态的内存区域,调用函数(calle ...
- Redis进阶知识一览
Redis的持久化机制 RDB: Redis DataBase 什么是RDB RDB∶每隔一段时间,把内存中的数据写入磁盘的临时文件,作为快照,恢复的时候把快照文件读进内存.如果宕机重启,那么内存里的 ...
- 【算法】归并排序(Merge Sort)(五)
归并排序(Merge Sort) 归并排序是建立在归并操作上的一种有效的排序算法.该算法是采用分治法(Divide and Conquer)的一个非常典型的应用.将已有序的子序列合并,得到完全有序的序 ...
- FreeMarker速查手册
一.开始 原理图 引入FreeMarker依赖 <dependency> <groupId>org.freemarker</groupId> <artifac ...
- fpn(feature-Pyramid-network)学习笔记
FPN(特征金字塔网络)学习笔记 论文 在物体检测里面,有限计算量情况下,网络的深度(对应到感受野)与 stride 通常是一对矛盾的东西,常用的网络结构对应的 stride 一般会比较大(如 32) ...