Redian新闻
>
掌握强大的 Git 变基命令 | Linux 中国

掌握强大的 Git 变基命令 | Linux 中国

科技
 
导读:学习如何使用 Git 来压扁、变基和精选。                                       
本文字数:4703,阅读时长大约:6分钟

学习如何使用 Git 来压扁、变基和精选。

当我与别人谈到 Git 时,几乎每个人都对 git rebase 命令🔗 opensource.com 有强烈的印象,这个命令让许多人遇到了问题,而不得不更改目录、删除仓库、然后再重新克隆一个仓库。我认为这是因为他们误解了分支是如何工作,遇到了一个非常糟糕的默认界面,还有一些合并冲突把事情搞得一团糟。

怎么找不到 git squash 命令?

如果你曾在本地的仓库提交过很多次,并希望能把这些提交都合并为一个提交,接下来,我们就来介绍能用什么 Git 命令达到这个目的。Git 称这个概念为 “压扁提交(squash commits)”。我在编写文档时发现了这个概念:我花了十几个提交才修改好我的 Markdown 文档,但是仓库的维护者不想看到我的所有尝试,以免扰乱了该项目的历史,所以我被告知“需要压扁你的提交”。

压扁提交听起来是一个很有用的方法。但是只有一个问题:我不知道该怎么做。作为 Git 的新手,我做了任何人会做的事情:我去查阅 git-squash 的手册,但我立即遇到了阻碍:

  1. $ man git-squash
  2. > No manual entry for git-squash

我发现没有一个名为 squash 的 Git 命令,而是被要求 运行一个完全独立的命令:git rebase 命令🔗 opensource.com,该命令能将我的所有提交最终合并为一个提交。

我知道我碰到一个常见的情形:已经使用工具一段时间的人使用了行话或引用了一个概念,这个概念对他们来说是非常清楚的,但对新手来说就不能明白了。从概念上讲,这个情况看起来是这样的:

Image of 6 bowls of different colored spices, and an arrow pointing to the second image of all the spices blended into one bowl.

我这样说是为了鼓励你,你绝对不是第一个或最后一个 被 Git 或谈论 Git 的人 弄糊涂的人。你可以要求对方说明白他的意见,并帮助你应该使用的正确命令。仓库的维护者实际上的意思是,“使用 git rebase 命令**,将很多提交压扁成一个提交”。

现在就来学习 git rebase 命令吧

git rebase 命令会将一个提交链从其第一个父级中删除,并将其放置在另一个提交链的末尾,将两个提交链组合成一个长链,而不是两个并行链。我意识到这是一个很复杂的定义。

回想一下 Git 的提交是如何链接在一起的,你可以看到,除了初始的 main(或 master)分支外,任何分支都有一个 父提交(parent commit) 作为该链的 “基础(base)”。“变基(rebase)” 能使另一个链中的最后一个提交成为指定分支的新 “基础提交(base commit)”。

在 Git 中整合来自不同分支的修改主要有两种方法:合并(merge) 以及 变基(rebase),你可能更熟悉 git merge 命令。接下来,就来看看 [git-scm.com🔗 git-scm.com] 是如何解释 git merge 和 git rebase 的差异:

Image of Git merge versus git rebase shown as numbered bubbles.

在合并示例中,它会把两个分支的最新快照(C3 和 C4)以及二者最近的共同祖先(C2)进行三方合并,合并的结果是生成一个新的快照(C5)。experiment 的分支指针仍然存在,仍然指向 C4

在变基示例中,它提取在 C4 中引入的补丁和修改,然后在 C3 的基础上应用一次,使 C3 成为 C4 的新父级,并产生了一个名为 C4' 的新提交。

(LCTT 译注:具体的命令如下:

  1. $ git checkout experiment
  2. $ git rebase main
  3. First, rewinding head to replay your work on top of it...
  4. Applying: added staged command

它的原理是首先找到这两个分支 —— 即当前分支 experiment、变基操作的目标基底分支 main —— 的最近共同祖先 C2,然后对比当前分支相对于该祖先的历次提交,提取相应的修改并存为临时文件,然后将当前分支指向目标基底 C3,最后以此将之前另存为临时文件的修改依序应用。)

值得注意的是,分支指针 main 没有移动。要让 Git 将指针移动到链的末尾(由experiment 指向),你还需要执行合并。

(LCTT 译注:具体的命令如下:

  1. $ git checkout main
  2. $ git merge experiment

master 分支的快进合并

此时,C4' 指向的快照就和上面使用 merge 命令的例子中 C5 指向的快照一模一样了。)

git rebase 并不能替代 git mergegit rebase 是一种用于制作更清晰的历史记录,以与 git merge 结合使用的工具。

(LCTT 译注:使用 git rebase 命令将提交到某一分支上的所有修改都移至另一分支上,就好像“重新播放”一样。)

交互式变基能给你一个更友好的界面!

从命令行执行 git rebase 命令,最可怕的地方在于它糟糕的默认界面。运行命令 git rebase <target-refr> 要么有效,要么会变得一团糟,因为它没有太多的反馈或方法来确保它做你想做的事情。幸运的是,git rebase 命令和许多其他 Git 命令一样,具有 交互模式(interactive mode),你可以使用参数 -i 或者 -interactive 来使用交互模式。

Image of the Git lens interactive Rebase tool in VS Code.

在使用交互式模式时,git rebase 会从一个糟糕的黑框界面转换为一个选项菜单,允许你选择对正在变基的提交链所做的事。对于每个提交,你可以选择

◈ 选用(pick):按原样包含
◈ 重写(reword):重写提交消息
◈ 编写(edit):在变基完成之前对提交中的文件进行进一步更改
◈ 压扁(squash):将多个提交压缩成一个提交,保留所有提交消息
◈ 修理(fixup):将多个提交压缩成一个提交,但只保留最后一个提交消息
◈ 丢弃(drop):丢弃此提交

就我个人而言,我更喜欢 VS Code 的开源 GitLens 扩展🔗 marketplace.visualstudio.com 使用下拉选择列表布局选项的方式,但 Git 允许你使用任何编辑器选择这些选项。对于 Emacs 或 Vim 等纯文本工具,你需要键入选择,而不是从菜单中选择,但最终结果仍然是相同的。

何时做变基

知道 何时 做变基与知道 如何 做变基同样重要。事实上,如果你不在乎你的仓库历史提交消息有点混乱的话,那么你可以永远都不使用 git rebase 命令。但是,如果你想要更干净的历史提交消息,并且想要更少扰乱你的图形视图的提交,那么当你使用 git rebase 命令时,有一个重要的经验法则需要时刻记住:

“不要变基你存储库以外的的提交,那些提交可能是别人工作的基础。”

如果你遵循该准则,不会发生什么大问题的。

简而言之,如果你让一个本地分支来完成你的工作,变基是没有问题的。但一旦该分支被 推送(push) 了,就不要再变基该分支了。当然,你想要怎么做完全取决于你自己。

希望你会认为上述内容有助于你理解 git rebase 命令的工作原理,并能让你更有信心地使用它。与任何 Git 命令一样,练习是学习和理解怎么做的唯一方法。我鼓励你勇敢地尝试 交互式变基(interactive rebase) git rebase -i <branch name>

接下来学习 Git cherry-pick 命令吧

大多数开发人员将修改提交到某一分支上,但是之后发现他们一直提交到了错误的分支上。理想情况下,他们可以拿走那个提交,然后把它移到正确的分支,这正是 git cherry-pick 命令的作用。

git cherry-pick 命令利用了变基单个提交的方法。这一用法非常常见,以至于有了它自己的命令。

Image of a woman picking a cherry from one tree and putting on another tree.

要使用 git cherry-pick,你只需告诉 Git 你要移动到“那个分支”的提交 ID(由 HEAD 指向):

  1. $ git cherry-pick <target-ref>

如果出现问题,你可以根据 Git 提供的错误消息,来进行恢复:

  1. $ git cherry-pick -i 2bc01cd
  2. Auto-merging README.md
  3. CONFLICT (content): Merge conflict in README.md
  4. error: could not apply 2bc01cd added EOF lines
  5. hint: After resolving the conflicts, mark them with
  6. hint: "git add/rm ", then run
  7. hint: "git cherry-pick --continue".
  8. hint: You can instead skip this commit with "git cherry-pick --skip".
  9. hint: To abort and get back to the state before "git cherry-pick",
  10. hint: run "git cherry-pick --abort".
  11. $ git cherry-pick --abort

让 Git 更强大

git rebase 命令是 Git 实用程序强大的地方之一。你最好在测试仓库中先练习一下怎么使用,一旦你熟悉了它的概念和工作流程,你就可以给仓库一个清晰历史消息记录了。


via: https://opensource.com/article/22/11/advanced-git-commands

作者:Dwayne McDaniel 选题:lkxed 译者:chai001125 校对:wxy

本文由 LCTT 原创编译,Linux中国 荣誉推出

LCTT 译者 :chai001125
🌟🌟🌟
翻译: 21.0 篇
|
贡献: 62 天
2022-10-06
2022-12-07
https://linux.cn/lctt/chai001125
欢迎遵照 CC-BY-SA 协议规定转载,
如需转载,请在文章下留言 “转载:公众号名称”,
我们将为您添加白名单,授权“转载文章时可以修改”。

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
如何通过 chroot 恢复 Arch Linux 安装 | Linux 中国如何在 Ubuntu Linux 上更新谷歌 Chrome | Linux 中国Linux 中的 su 和 sudo 命令有什么区别? | Linux 中国如何在 Ubuntu 等 Linux 中安装 Python 3.11 | Linux 中国13 个从头开始构建的独立 Linux 发行版 | Linux 中国如何提高 Ubuntu 和其他 Linux 系统中的扬声器音量 | Linux 中国如何在 Linux 中确定运行的是那种初始化系统 | Linux 中国解决 Linux 中的 “Bash: Command Not Found” 报错 | Linux 中国Acciona Energía 收购德州最大的电池储能项目Linux Mint 的更新管理器现在支持 Flatpak | Linux 中国12 个对新手最重要的 Linux 命令 | Linux 中国伊洛拉峡谷如何在 Arch Linux 中启用 Snap 支持 | Linux 中国如何在 Ubuntu 和其他相关 Linux 中安装 Python 3.10 | Linux 中国Arch Linux 中用于包管理的图形化应用 | Linux 中国使用 PSCP 将文件和文件夹从 Windows 传输到 Linux | Linux 中国LURE 初窥!将 AUR 带入所有 Linux 发行版 | Linux 中国在 Linux 中如何从命令行查找默认网关的 IP 地址 | Linux 中国在 Linux 中创建 LVM 分区的分步指南 | Linux 中国新冠病毒突变基本饱和?高福、张文宏这样预判Rhino Linux:滚动发布但也很稳定的 Ubuntu | Linux 中国打造万圣节 Linux 桌面 | Linux 中国Rosalía 登意大利版《VOGUE》封面!伊斯坦布爾轉機快閃遊准备好在 Debian Linux 上获得 Ubuntu MATE 的体验吧! | Linux 中国如何在 Ubuntu 和其他 Linux 中检查 CPU 和硬盘温度 | Linux 中国Woodbridge 秋会,Humber 河【了不起的妈妈系列】Yan通过 SSH 在远程 Linux 系统上执行命令 | Linux 中国如何在 Silverblue 上变基到 Fedora Linux 37 | Linux 中国联航(UA)新气象如何在最小安装的 CentOS、RHEL、Rocky Linux 中设置互联网 | Linux 中国在你的 Linux 终端中玩经典的贪吃蛇游戏 | Linux 中国使用这个多功能的 Linux 命令转换音频文件 | Linux 中国开源朗读者:使用 Linux 的优势和劣势 | Linux 中国
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。