git 优雅的撤销中间某次提交方法

环境

git : 2+

前言

最近两天,公司的git合并代码时,出现了严重的问题,浪费很多时间;
现在记录下;
情况是这样的,一个同事自己的本地分支(远程没有),不知怎么的,有了别人开发分支的代码,而他自己又不知道;
其在切换到主分支,并merge自己的分支,此时其已经把别人正在开发的代码都合并到了主分支。

到了晚上准备升级时,才发现,主分支的代码出了问题;此时版本库是这样的:

如图 100047dcc这一步就有不该有的代码;

而此时版本库已经提交过了很多次,现在的问题就是,如何撤销掉100047dcc提交的代码,并且保留其他人提交的代码。

这个问题,折腾到了晚上9点半左右,尝试了网上给出的:

git rebase -i commit_id
//再通过将pick改为drop

但是,实际的效果是,100047dcc代码没了,其他人提交的代码也没有了!
也就是给人感觉和git reset --hard be8c6f6dd没有什么区别!

最后因为太晚,从提交记录上看,100047dcc之后就一个人提交了代码,所以就执行了:

//先切一个备份分支
git branch -b master_tmp
//再执行
git reset --hard be8c6f6dd

之后,那个人(也就是我)从备份分支上把文件拷贝回来—(因为我是直接在主分支上改的,自己的分支并没有代码)。

第二天,我打算去拷贝文件,我执行如下操作:

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master)
$ git pull
remote: Counting objects: 44, done.
remote: Compressing objects: 100% (23/23), done.
remote: Total 26 (delta 19), reused 0 (delta 0)
Unpacking objects: 100% (26/26), done.
From gitlab.gofund.cn:gg-service/ggservice
 + 1784b12...384decc master   -> origin/master (forced update)
  f8f2b19..eb33489 devyaomy  -> origin/devyaomy
 * [new branch]   master_tmp -> origin/master_tmp
Already up-to-date.

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master)
$ git status
On branch master
Your branch is ahead of 'origin/master' by 796 commits.
 (use "git push" to publish your local commits)
nothing to commit, working directory clean

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master)
$ git push
Total 0 (delta 0), reused 0 (delta 0)
To git@gitlab.gofund.cn:gg-service/ggservice.git
  384decc..1784b12 master -> master

简单的说,我的操作就是两步:
1git pull
2、git push
结果又把代码还原回去了!

为啥呢?

虽然昨天晚上,把远程库的版本回退到了正确的版本,但是我的本地主分支还是最新的commit,也就是说,相比远程库,我本地库是超前了多次提交,毕竟因为远程库回退了嘛!

这个时候,我必须也得对本地库进行回退,回退到线上相同的commit节点才行。

这个时候,我做了以下几个操作:

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master)
$ git reset --soft 384deccaa6
$ git status
On branch master
Your branch is up-to-date with 'origin/master'.
Changes to be committed:
 (use "git reset HEAD <file>..." to unstage)

    modified:  .gitignore
    modified:  conf/application.conf
    new file:  conf/hanlp.properties
    new file:  dataservice/app/ggservice/common/UserCodeEnum.java
    new file:  dataservice/app/ggservice/v1/email/action/BindEmailGG3Action.java
    new file:  dataservice/app/ggservice/v1/email/action/SendEmailCaptchaGG3Action.java
    // 文件太多不一一显示
yutao@yutao MINGW64 /d/sts/workspace/ggservice (master)
$ git reset --hard 384deccaa6

上面敲了很多命令,其实真正只需要git reset --hard 384deccaa6即可。

git reset --hard 384deccaa6

接下来,我开始复制粘贴,从备份分支上,把代码拷贝下。
真的操蛋,这等于是增加工作量啊!

revert 撤销某次提交

到了下午,又有个同事干了类似我上午的操作。把不该有的代码提交上去了!
这就麻烦了,虽然远程库回退了!结果是要求凡是pull最新代码的人,都得进行本地回退的操作。
否则,就没完没了!

到了晚上,对着备份分支进行测试,终于找到了优雅的解决办法!

这就是revert命令

该命令就是为撤销某次提交而存在的;

首先,我是明白100047dcc这次提交是有问题的,这是问题的源头;
也就是说,只要我们把这次提交的给撤销了,就不会有问题了!

步骤 一

$ git revert 100047dcc
error: Commit 100047dccb58f3ee5e27b0dfaf5c02ac91dc2c73 is a merge but no -m option was given.
fatal: revert failed

结果报错了,报了一个Commit is a merge but no -m option was given.

为什么呢?

如果100047dcc这只是一个普通的提交,其实是不会报错的!
但是,这是一个merge的提交。

那么在撤销时,git并不知道我要撤销具体哪次!如下图:

这个时候,怎么办呢?

我的做法

步骤二

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp)
$ git revert 100047dcc -m 1
error: could not revert 100047d... Merge branch 'master' of gitlab.gofund.cn:gg-service/ggservice into wjs
hint: after resolving the conflicts, mark the corrected paths
hint: with 'git add <paths>' or 'git rm <paths>'
hint: and commit the result with 'git commit'

我执行了这样的一个操作:

git revert 100047dcc -m 1

参数 -m 就是指定要撤销的那个提价,从左往右,从1开始数;也就是我撤销的是ca4a7ff999

接着其把代码冲突,然后我就解决冲突,保留主分支的代码,去掉那个人的代码。

解决完冲突后,我执行如下操作:

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git add -A

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git status
On branch master_tmp
Your branch is up-to-date with 'origin/master_tmp'.
You are currently reverting commit 100047d.
 (all conflicts fixed: run "git revert --continue")
 (use "git revert --abort" to cancel the revert operation)

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

    modified:  dataservice/app/ggservice/v1/datacentre/action/GetIncomeDistributeAction.java
    //文件太多省略。。。

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git commit -m "ceshi"
[master_tmp d2ae829] ceshi
 18 files changed, 95 insertions(+), 396 deletions(-)

我上面执行的语句其实就是:

$ git add -A
$ git commit -m "ceshi"

步骤三

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp)
$ git revert 100047dcc -m 2
error: could not revert 100047d... Merge branch 'master' of gitlab.gofund.cn:gg-service/ggservice into wjs
hint: after resolving the conflicts, mark the corrected paths
hint: with 'git add <paths>' or 'git rm <paths>'
hint: and commit the result with 'git commit'

也就是执行:

$ git revert 100047dcc -m 2

即 撤销be8c6f6dde的提交,这个时候也会提示代码冲突了,

接着和上面一样,解决冲突,在提交:

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git status
On branch master_tmp
Your branch is ahead of 'origin/master_tmp' by 1 commit.
 (use "git push" to publish your local commits)
You are currently reverting commit 100047d.
 (fix conflicts and run "git revert --continue")
 (use "git revert --abort" to cancel the revert operation)

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

    modified:  .gitignore
    deleted:  conf/hanlp.properties
    deleted:  dataservice/app/ggservice/common/UserCodeEnum.java
    deleted:  dataservice/app/ggservice/v1/email/action/BindEmailGG3Action.java
    deleted:  dataservice/app/ggservice/v1/email/action/SendEmailCaptchaGG3Action.java
    deleted:  dataservice/app/ggservice/v1/email/service/EmailCaptchaService.java
    deleted:  dataservice/app/ggservice/v1/expert/action/GetExpertOfStockAssessAction.java
    modified:  dataservice/app/ggservice/v1/expert/service/ExpertGG3Service.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/action/GetMyStockLabelInfoAction.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/action/UpdateUserTokenInfoAction.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/service/AppDOSInfoService.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/service/UserInfoService.java
    modified:  dataservice/app/ggservice/v1/graph/action/GetStockPlateComponentAction.java
    modified:  dataservice/app/ggservice/v1/graph/service/StockPlateService.java
    // 文件太多省略。。。

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git add -A

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git status
On branch master_tmp
Your branch is ahead of 'origin/master_tmp' by 1 commit.
 (use "git push" to publish your local commits)
You are currently reverting commit 100047d.
 (all conflicts fixed: run "git revert --continue")
 (use "git revert --abort" to cancel the revert operation)

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

    modified:  .gitignore
    modified:  conf/application.conf
    deleted:  conf/hanlp.properties
    deleted:  dataservice/app/ggservice/common/UserCodeEnum.java
    deleted:  dataservice/app/ggservice/v1/email/action/BindEmailGG3Action.java
    deleted:  dataservice/app/ggservice/v1/email/action/SendEmailCaptchaGG3Action.java
    deleted:  dataservice/app/ggservice/v1/email/service/EmailCaptchaService.java
    deleted:  dataservice/app/ggservice/v1/expert/action/GetExpertOfStockAssessAction.java
    modified:  dataservice/app/ggservice/v1/expert/service/ExpertGG3Service.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/action/GetMyStockLabelInfoAction.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/action/UpdateUserTokenInfoAction.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/service/AppDOSInfoService.java
    modified:  dataservice/app/ggservice/v1/ggmtoolbox/service/UserInfoService.java
    modified:  dataservice/app/ggservice/v1/graph/action/GetStockPlateComponentAction.java
    modified:  dataservice/app/ggservice/v1/graph/service/StockPlateService.java
    modified:  dataservice/app/ggservice/v1/hq/action/GetStockHistoryDynamicAction.java
    modified:  dataservice/app/ggservice/v1/keybordspirit/action/GetMyGroupStockIndexAction.java

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp|REVERTING)
$ git commit -m "使用revert 版本号 -m 1或者2同时进行撤销某次提交测试"
[master_tmp 236da00] 使用revert 版本号 -m 1或者2同时进行撤销某次提交测试
 95 files changed, 2093 insertions(+), 10011 deletions(-)
 delete mode 100644 conf/hanlp.properties
 delete mode 100644 dataservice/app/ggservice/common/UserCodeEnum.java
 delete mode 100644 dataservice/app/ggservice/v1/email/action/BindEmailGG3Action.java
 delete mode 100644 dataservice/app/ggservice/v1/email/action/SendEmailCaptchaGG3Action.java
 delete mode 100644 dataservice/app/ggservice/v1/email/service/EmailCaptchaService.java
 delete mode 100644 dataservice/app/ggservice/v1/expert/action/GetExpertOfStockAssessAction.java
 delete mode 100644 dataservice/app/ggservice/v1/mobile/action/BindMobileGG3Action.java
 delete mode 100644 dataservice/app/ggservice/v1/mobile/action/SendMobileCaptchaGG3Action.java
 rewrite dataservice/app/ggservice/v1/mystocktags/service/MyStockTagService.java (82%)
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetAuthorRankListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetAuthorRecommendReportListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetHonoraryAuthorListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetHotIndustryListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetHotStockListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetHotThemeListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/action/GetOrganRankListAction.java
 delete mode 100644 dataservice/app/ggservice/v1/report/condition/AuthorOrganRankCondition.java
 delete mode 100644 dataservice/app/ggservice/v1/report/condition/HotReportCondition.java
 delete mode 100644 dataservice/app/ggservice/v1/report/service/HotReportService.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/AutoLoginAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/BindOuterChannelAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/EmailRegisterAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/GetUserAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/IsAccountExistAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/LoginAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/LogoutAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/OuterChannelLoginAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/action/RegisterAction.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/service/LoginService.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/service/RegisterService.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/service/UserCommonUtils.java
 delete mode 100644 dataservice/app/ggservice/v1/usergg/service/UserService.java

即:

$ git add -A
$ git commit -m "使用revert 版本号 -m 1或者2同时进行撤销某次提交测试"

可以看出删除掉了那个人提交的文件。

最后一步

yutao@yutao MINGW64 /d/sts/workspace/ggservice (master_tmp)
$ git push

这样就把那个人提交错误的代码给删除了,其他人的本地分支也不需要版本回退了!
一次改好,到处OK!

总结

当想撤销中间某次提交时,强烈建议使用revert命令,而不是reset
git reset –hard commit_id 虽然可以回退远程库,但是其要求pull最新代码的每个人的本地分支都要进行版本回退。这样就增加工作量了!

正确的步骤:

git revert commit_id
//如果commit_id是merge节点的话,-m是指定具体哪个提交点
git revert commit_id -m 1
//接着就是解决冲突
git add -A
git commit -m ".."
git revert commit_id -m 2
//接着就是解决冲突
git add -A
git commit -m ".."
git push

其中git revert commit_id -m 数字是针对,merge提交点的操作。
如果是普通的提交点,不需要这么麻烦。

参考地址:
[Git高级教程(二)] 远程仓库版本回退方法
https://www.cnblogs.com/ShaYeBlog/p/5368064.html
https://blog.csdn.net/hongchangfirst/article/details/49472913

到此这篇关于git 优雅的撤销中间某次提交方法的文章就介绍到这了,更多相关git 撤销某次提交内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Git 撤销操作、删除文件和恢复文件

    大致介绍 经过前面的学习,已经建立了版本库,并上传了文件,这次来学习对这些文件进行基本的操作,即: ◆ 撤销操作 ◆ 删除文件 ◆ 恢复文件 我在此之前,已经将三个文件提交到了版本库 撤销操作 撤销操作的语法: git checkout -- 文件名 撤销操作一般有两种情况: ◆ 文件修改后还没有提交到暂存区,进行撤销操作之后,文件恢复到和版本库中一模一样 ◆文件修改后提交到了暂存区,进行撤销之后,文件恢复到在提交到暂存区之前的状态 现在index.htm中的内容是: index.html 我们

  • git rebase 成功之后撤销的操作方法

    rebase在git中是一个非常有魅力的命令,使用得当会极大提高自己的工作效率:相反,如果乱用,会给团队中其他人带来麻烦.它的作用简要概括为:可以对某一段线性提交历史进行编辑.删除.复制.粘贴:因此,合理使用rebase命令可以使我们的提交历史干净.简洁! 下面看下git rebase 成功之后如何撤销,感兴趣朋友看下. git rebase 过程中可以使用git --abort/--continue来进行操作,成功之后如何撤销呢? 首先执行git reflog查看本地记录 可知本次rebase

  • 基于Git的常用撤销技巧与解决冲突方法(推荐)

    git checkout . #本地所有修改的.没有的提交的,都返回到原来的状态 git stash #把所有没有提交的修改暂存到stash里面.可用git stash pop回复. git reset --hard HASH #返回到某个节点,不保留修改. git reset --soft HASH #返回到某个节点.保留修改 撤销Git add操作 git reset HEAD <file> # 取消add操作并保留修改 git checkout -- <file> # 若继续

  • Git撤销&回滚操作(git reset 和 get revert)

    git的工作流 工作区:即自己当前分支所修改的代码,git add xx 之前的!不包括 git add xx 和 git commit xxx 之后的. 暂存区:已经 git add xxx 进去,且未 git commit xxx 的. 本地分支:已经git commit -m xxx 提交到本地分支的. 代码回滚 在上传代码到远程仓库的时候,不免会出现问题,任何过程都有可能要回滚代码: 1.在工作区的代码 git checkout -- a.txt # 丢弃某个文件,或者 git chec

  • git如何撤销commit的方法(未push)

    撤销commit一般用git reset ,语法如下: git reset [ --mixed | --soft | --hard] [<commit ID>] 1.使用参数--mixed(默认参数),如git reset --mixed <commit ID>或git reset <commit ID> 撤销git commit,撤销git add,保留编辑器改动代码 2.使用参数--soft,如git reset --soft<commit ID> 撤销

  • Git使用小技巧之回滚与撤销详解

    前言 日常的开发,我们难免会创建错误的git提交记录,整个时候git给我们提供了两个命令来解决这个问题.一个命令是git reset,另一个是git revert.两者有啥区别呢?两者主要的区别是,git reset命令是回滚某次提交,被回滚的提交将不会出现在提交记录中,而git revert命令是创建一个新的提交来达到撤销的目的,被撤销的提交和撤销的提交都会出现在提交记录中. 准备 首先呢,我们还是需要准备一个git仓库,并且准备一个可以修改的文件,文件名无所谓,我这里就创建了一个README

  • Git里多种撤销操作的最佳方法

    前言 相信大家都知道任何版本控制系统的一个最有的用特性就是"撤销 (undo)"你的错误操作的能力.在 Git 里,"撤销" 蕴含了不少略有差别的功能.当你进行一次新的提交的时候,Git 会保存你代码库在那个特定时间点的快照:之后,你可以利用 Git 返回到你的项目的一个早期版本. 撤销一个"已公开"的改变 场景: 你已经执行了 git push, 把你的修改发送到了 GitHub,现在你意识到这些 commit 的其中一个是有问题的,你需要撤销

  • git 优雅的撤销中间某次提交方法

    环境 git : 2+ 前言 最近两天,公司的git合并代码时,出现了严重的问题,浪费很多时间: 现在记录下: 情况是这样的,一个同事自己的本地分支(远程没有),不知怎么的,有了别人开发分支的代码,而他自己又不知道: 其在切换到主分支,并merge自己的分支,此时其已经把别人正在开发的代码都合并到了主分支. 到了晚上准备升级时,才发现,主分支的代码出了问题:此时版本库是这样的: 如图 100047dcc这一步就有不该有的代码: 而此时版本库已经提交过了很多次,现在的问题就是,如何撤销掉10004

  • 如何使用Git优雅的回滚实现

    开发过程中,我们经常会遇到代码回滚的情况.正常人都知道,git 回滚有两大宝: git revert git reset 当我们在本地开发,还未 git push 到远端时,可以毫无顾忌的使用 git reset 进行回滚.更多的情况中,我们不仅 push 了,而且由于开发周期长,在开发过程中不断的 merge master 和 merge other-branch 以发布到预发环境测试或者多需求合并测试. 突然 上线后用户投诉,需要马上下线本次需求中的 A.B.C,只保留 D.E.F,迅速回滚

  • Git创建子分支,合并分支并提交

    Git合并分支后,需要将子分支提交到git仓库,这个时候就需要单独提交子分支,其步骤如下: 1.先创建子分支,并包含最新当前分支下的修改数据 git checkout -b sonBranch 2.将新分支内容全部提交 git add . git commit -m "完成开发" 3.在码云创建一个子分支 git push -u origin sonBranch 4.切换到主分支 git checkout master 5.合并分支,更新主分支内容 git merge sonBranc

  • 详解git submodule update获取不到最新提交的代码

    今天遇到了一个git submodule update获取不到最新代码的问题,闹了半天原来是自己对git submodule理解不到位引起的.解决了后,感觉虽然没有什么高深的地方,但是不清楚的时候还是需要费时间去查找问题,所以还是在此记录如下. 1.问题描述 同事在共同使用的一个工程里修改了代码,这个工程在我本地的工程中是作为submodule来使用的,但是我在本地执行以下命令却获取不到最新提交的代码. git submodule update --init --recursive 2.问题原因

  • 详解git的分支与合并的两种方法

    如何将两个分支合并到一起.就是说我们新建一个分支,在其上开发某个新功能,开发完成后再合并回主线. 1.   git merge 咱们先来看一下第一种方法 -- git merge 在 Git 中合并两个分支时会产生一个特殊的提交记录,它有两个父节点.翻译成自然语言相当于:"我要把这两个父节点本身及它们所有的祖先都包含进来."下面具体解释. # 创建新分支 bugFix git branch bugFix # 切换到该分支 git checkout bugFix # 提交一次 git c

  • ahooks解决用户多次提交方法示例

    目录 引言 场景 useLockFn 缺点 axios 自动取消重复请求 axios 取消请求 如何自动取消重复的请求 思考与总结 引言 本文是深入浅出 ahooks 源码系列文章的第四篇,这个系列的目标主要有以下几点: 加深对 React hooks 的理解. 学习如何抽象自定义 hooks.构建属于自己的 React hooks 工具库. 培养阅读学习源码的习惯,工具库是一个对源码阅读不错的选择. 注:本系列对 ahooks 的源码解析是基于 v3.3.13.自己 folk 了一份源码,主要

  • JavaScript在form表单中使用button按钮实现submit提交方法

    submit是button的一个特例,也是button的一种,它把提交这个动作自动集成了,submit和button,二者都以按钮的形式展现,看起来都是按钮,所不同的是type属性和处发响应的事件上. 在javaScript中关于submit和button的区别介绍 在form表单提交中,使用button来间接完成submit的提交更为灵活.下面是实现方法: 使用button按钮实现submit提交,需要在button标签中的使用onclick方法,然后在JavaScript中实现具体,代码如下

  • Git发现git push origin master 报错的解决方法

    git push origin master 报错的解决方法,分享给大家,具体如下: 错误提示如下 [root@linux1 php]# git push -u origin master To git@github.com:kangvcar/Results-Systems--PHP.git ! [rejected] master -> master (fetch first) error: failed to push some refs to 'git@github.com:kangvcar

随机推荐