[git] 如何丢弃对一个文件的改动?

背景

当我们进行日常开发时,有时候需要丢弃对一个文件的改动。为了便于描述,我们假设这个文件是 f.txt \text{f.txt} f.txt。会有两种情况 ⬇️

  • 仅丢弃 f.txt \text{f.txt} f.txt 尚未暂存 的改动
  • 丢弃 f.txt \text{f.txt} f.txt 已暂存 的改动 + 尚未暂存 的改动

本文会对这两种情况分别进行介绍

要点

仅丢弃尚未暂存的改动

执行以下两个命令中的 任何一个 都可以丢弃 尚未暂存 的改动

bash 复制代码
git restore <file>
## 或者
git checkout <file>

注意事项

  • 执行命令时,需要将 <file> 替换为您想指定的文件(例如 f.txt \text{f.txt} f.txt)
  • 如果您是第一次使用它们,请不要用重要的项目来进行试验,以防误操作(尚未暂存 的改动被丢弃之后就找不回来了)

丢弃 已暂存 的改动 + 尚未暂存 的改动

执行以下两个命令中的 任何一个 都可以丢弃 已暂存 的改动 + 尚未暂存 的改动

bash 复制代码
git restore --staged --worktree <file>
## 或者
git checkout HEAD <file>

注意事项

  • 执行命令时,需要将 <file> 替换为您想指定的文件(例如 f.txt \text{f.txt} f.txt)
  • 如果您是第一次使用它们,请不要用重要的项目来进行试验,以防误操作(已暂存/尚未暂存 的改动被丢弃之后就找不回来了)

正文

示例

准备工作

请将以下代码保存为 main.sh (具体的名称并不重要)

bash 复制代码
## Prepare a temp dir for our test
mkdir temp
cd temp

## Prepare for the initial commit
git init
echo "Content commited" >> f.txt
git add f.txt
git commit -m "initial commit"

## Stage some content in f.txt
echo "Staged content in f.txt" >> f.txt
git add f.txt

## Add some content to be staged in f.txt
echo "Content to be staged ..." >> f.txt

通过执行下方的命令就可以运行 main.sh ⬇️

bash 复制代码
bash main.sh

运行完 main.sh 之后,当前目录中会多出一个 temp 目录。在当前目录执行 tree temp 命令后,应该可以看到如下的内容

text 复制代码
temp
└── f.txt

1 directory, 1 file

通过执行如下命令,可以切换到 temp 目录中 ⬇️

bash 复制代码
cd temp

temp 目录中,我们执行如下命令就可以看到 f.txt \text{f.txt} f.txt 的状态

bash 复制代码
git status

执行该命令后,可以看到 f.txt \text{f.txt} f.txt 中既有 已暂存 的改动(如下图绿色框所示),也有 尚未暂存 的改动(如下图红色框所示)

各个状态的 f.txt \text{f.txt} f.txt 的内容列举如下 ⬇️

内容
commit \text{commit} commit 的 f.txt \text{f.txt} f.txt Content commited
已暂存的 f.txt \text{f.txt} f.txt Content commited Staged content in f.txt
当前的 f.txt \text{f.txt} f.txt Content commited Staged content in f.txt Content to be staged ...

如果我们想让 f.txt \text{f.txt} f.txt 变回下图红框里的这个样子,那么只需要丢弃 尚未暂存 的改动

如果我们想让 f.txt \text{f.txt} f.txt 变回下图绿框里的这个样子,那么需要丢弃 已暂存 的改动 + 尚未暂存 的改动

我们分别来看

情形一:仅丢弃尚未暂存的改动

请注意:尚未暂存 的改动被丢弃之后就找不回来了

第一种方式:基于 git restore \text{git restore} git restore 命令

我们可以用 git restore \text{git restore} git restore 命令来丢弃 f.txt \text{f.txt} f.txt 中尚未暂存的改动 ⬇️

bash 复制代码
git restore f.txt

执行该命令后,我们再用 git status \text{git status} git status 命令和 cat f.txt \text{cat f.txt} cat f.txt 命令验证一下 ⬇️

从红色框的内容可以看出, f.txt \text{f.txt} f.txt 的变化符合预期。

第二种方式:基于 git checkout \text{git checkout} git checkout 命令

我们也可以用 git checkout \text{git checkout} git checkout 命令来丢弃 f.txt \text{f.txt} f.txt 中尚未暂存的改动 ⬇️

bash 复制代码
git checkout f.txt

执行该命令后,可以再用 git status \text{git status} git status 命令和 cat f.txt \text{cat f.txt} cat f.txt 命令验证一下 ⬇️

从红色框的内容可以看出, f.txt \text{f.txt} f.txt 的变化符合预期。

情形二:丢弃 已暂存 与 尚未暂存 的改动

请注意:已暂存/尚未暂存 的改动被丢弃之后就找不回来了

第一种方式:基于 git restore \text{git restore} git restore 命令

我们可以用 git restore \text{git restore} git restore 命令来丢弃 f.txt \text{f.txt} f.txt 中 已暂存 的改动 + 尚未暂存 的改动 ⬇️

bash 复制代码
git restore --staged --worktree f.txt

执行该命令后,我们再用 git status \text{git status} git status 命令和 cat f.txt \text{cat f.txt} cat f.txt 命令验证一下 ⬇️

从红色框的内容可以看出, f.txt \text{f.txt} f.txt 的变化符合预期。

第二种方式:基于 git checkout \text{git checkout} git checkout 命令

我们也可以用 git checkout \text{git checkout} git checkout 命令来丢弃 f.txt \text{f.txt} f.txt 中 已暂存 的改动 + 尚未暂存 的改动 ⬇️

bash 复制代码
git checkout HEAD f.txt

执行该命令后,我们再用 git status \text{git status} git status 命令和 cat f.txt \text{cat f.txt} cat f.txt 命令验证一下 ⬇️

从红色框的内容可以看出, f.txt \text{f.txt} f.txt 的变化符合预期。

参考资料

Git Cheat Sheet 中的以下内容

相关推荐
GoGeekBaird2 小时前
从 Prompt Engineering 到 Loop Engineering,我觉得 AI 开发这事儿终于开始变味了
后端·github
一条泥憨鱼3 小时前
【Redis】数据类型和常用命令
java·数据库·redis·后端·缓存
Oneslide4 小时前
初始化微信小程序
后端
hboot4 小时前
AI工程师第一课 - Python
前端·后端·python
阿正的梦工坊5 小时前
【Rust】12-借用检查器与非词法生命周期
开发语言·后端·rust
飞天狗1115 小时前
零基础JavaWeb入门——第2课:让网页“活”起来 —— JSP是什么?
java·开发语言·前端·后端·web
梦@_@境6 小时前
面向 Spring Boot 的可观测业务流程编排引擎
java·spring boot·后端
JAVA面经实录9177 小时前
Netty 全套系统化学习文档(零基础到高阶面试完整版)
java·后端
GetcharZp7 小时前
C++ 程序员的终极减负:仅需一个头文件,优雅搞定 HTTP 客户端与服务端
后端
IT_陈寒7 小时前
Python的pickle让我半夜加班,这破玩意儿太坑了
前端·人工智能·后端