Git 中糟糕的想法
本文改编自一个每年都被 GitHub Universe 忠实拒绝的演讲。我不明白为什么......
不管好坏,git 已经成为开源社区最普遍的工具之一。它可以帮助你有效地管理代码。它可以帮助远隔千里的工程师们互相协作。其核心非常简单,这就是为什么这么多博客文章中的图表不可避免地看起来都像图 1 中所示的那样。
图 1. Git 模型 (来源: https://nvie.com)
鲜少被详细讨论的不幸真相是 git 有其阴暗面:它会让我们觉得自己很蠢。我不在乎你是谁——我们都会遇到一个时刻,耸耸肩,放弃,然后赶紧去 Stack Overflow(座右铭:“此主题已关闭,因为它与主题无关”)上查找最佳方法,以摆脱我们自己造成的糟糕境地。唯一的问题是你会在疯狂吞噬你之前,或者在你开始靠养山羊为生之前,能在这条兔子洞里走多远。
究其核心,git 所做的就是跟踪文件和文件夹的更改。git commit
有效地在给定时间点拍摄文件系统的快照(以添加到暂存区的项目为代表)
cquinn@1d732dc08938 ~/demo1 % git init
Initialized empty Git repository in /home/cquinn/demo1/.git/
cquinn@1d732dc08938(master|...) ~/demo1 % git add ubuntu.iso
cquinn@1d732dc08938(master|·1) ~/demo1 % git commit
↪-m "Initial commit"
[master (root-commit) b0d3bfb] Initial commit
1 file changed, 0 insertions(+), 0 deletions(-)
create mode 100644 ubuntu.iso
cquinn@1d732dc08938(master|✓) ~/demo1 % git rm --cached
↪ubuntu.iso
rm 'ubuntu.iso'
cquinn@1d732dc08938(master|·1✓) ~/demo1 % git
↪commit -m "There I fixed it"
[master 2d86934] There I fixed it
1 file changed, 0 insertions(+), 0 deletions(-)
delete mode 100644 ubuntu.iso
cquinn@1d732dc08938(master|...) ~/demo1 % du -hs .git
174M .git
因此,如果你做了一些愚蠢的事情,例如提交大型二进制文件,你不能仅仅回滚提交——它仍然会存在于你的 git 仓库中。如果你已经将该内容推送到其他地方,你就需要强制重写历史记录,可以使用 git-filter-branch
或 bfg
。无论哪种方式,这都会给与你共享仓库的其他人带来额外的不便。
从根本上说,git 所做的就是在仓库的顶层创建一个 .git 文件夹。这个子目录包含随时间变化的文件和文件夹。等等,难道没有工具可以做到这一点吗?
cquinn@1d732dc08938 ~/demo2 % git init
Initialized empty Git repository in /home/cquinn/demo2/.git/
cquinn@1d732dc08938(master|✓) ~/demo2 % cd .git
cquinn@1d732dc08938 ~/demo2/.git % ls
HEAD branches config description hooks info objects refs
cquinn@1d732dc08938 ~/demo2/.git % git init
Initialized empty Git repository in /home/cquinn/demo2/
↪.git/.git/
我不确定你会 为什么要做这样的事情,但重点是你肯定可以这样做。
cquinn@1d732dc08938 ~/demo2 % git
你有没有试过开始输入一个 git 命令,然后在谷歌搜索时迷失了方向?然后你找到了命令并粘贴进去
cquinn@1d732dc08938 ~/demo2 % git git status git: 'git' is not
a git command. See 'git --help'.
Did you mean this?
ign
然后你觉得自己很蠢。让我们解决这个问题
# echo git @? \> /usr/local/bin/git-git
cquinn@1d732dc08938(master|✓) ~/demo2 % sudo bash
root@1d732dc08938:~/demo2# echo 'git $@' >
↪/usr/local/bin/git-git
root@1d732dc08938:~/demo2# chmod +x /usr/local/bin/git-git
好了,就这样
cquinn@1d732dc08938(master|&#check;) ~/demo2 % git git git git
↪git git git status
On branch master
Initial commit
nothing to commit (create/copy files and use "git add" to track)
我不是说这是一个好主意——只是说它可以做到。
真正有用的 Git 技巧曾经在语法上搞砸过吗?
cquinn@1d732dc08938(master|·1) ~/demo4 % git stts
git: 'stts' is not a git command. See 'git --help'.
Did you mean this?
status
然后你坐在那里为自己感到难过。欢呼吧:git 具有自动纠错设置
cquinn@1d732dc08938(master|·1) ~/demo4 % git config
↪--global help.autocorrect 8
cquinn@1d732dc08938(master|·1) ~/demo4 % git stts
WARNING: You called a Git command named 'stts',
↪which does not exist.
Continuing under the assumption that you meant 'status'
in 0.8 seconds automatically...
On branch master
Your branch is up-to-date with 'origin/master'.
Changes to be committed:
(use "git reset HEAD <file>..." to unstage)
new file: file2.txt
--global
标签将此应用于用户的 ~/.gitconfig,但更相关的部分是最后两个参数。help.autocorrect
启用自动错别字检测,而 8
是你必须在几十分之一秒内按下 Ctrl-C 的时间,之后 git 才会对你的环境做一些可怕的事情。
我想指出的最后一个工具是 myrepos
。此工具允许你有效地操作构成你公司糟糕的微服务架构的众多仓库(有趣的事实:微服务最初是在一次会议演讲中作为一个玩笑提出的,但人们却当真了)
cquinn@Quinnversion ~ % mr list 31384 10:48:52
↪Fri 06- 8-2018
mr list: /Users/cquinn/.config/vcsh/repo.d/gitconfig.git
mr list: /Users/cquinn/.config/vcsh/repo.d/mr.git
mr list: /Users/cquinn/.config/vcsh/repo.d/ssh.git
mr list: /Users/cquinn/.config/vcsh/repo.d/tmux.git
mr list: /Users/cquinn/.config/vcsh/repo.d/vim.git
mr list: /Users/cquinn/.config/vcsh/repo.d/zsh.git
mr list: /Users/cquinn/Dropbox/src/docPR/complete/
↪amazon-cloud-directory-developer-guide
mr list: finished (7 ok)
cquinn@Quinnversion ~ % mr status 31385 10:48:53
↪Fri 06- 8-2018
mr status: /Users/cquinn/.config/vcsh/repo.d/gitconfig.git
On branch master
Your branch is up to date with 'origin/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: ../../../../.gitconfig
no changes added to commit (use "git add" and/or
↪"git commit -a")
mr status: /Users/cquinn/.config/vcsh/repo.d/mr.git
On branch master
Your branch is up to date with 'origin/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: ../../../../.mrconfig
no changes added to commit (use "git add" and/or
↪"git commit -a")
mr status: /Users/cquinn/.config/vcsh/repo.d/ssh.git
mr status: /Users/cquinn/.config/vcsh/repo.d/tmux.git
On branch master
Your branch is up to date with 'origin/master'.
nothing to commit, working tree clean
mr status: /Users/cquinn/.config/vcsh/repo.d/vim.git
mr status: /Users/cquinn/.config/vcsh/repo.d/zsh.git
Behind origin/master by 1 commits
M ../../../../.zsh/functions/gitstatus.py
M ../../../../.zshrc
mr status: /Users/cquinn/Dropbox/src/docPR/complete/
↪amazon-cloud-directory-developer-guide
mr status: finished (7 ok)
它还接受并发参数来并行化工作负载;例如,mr -j8 status
会产生八个工作线程。
还有两个临别提示给你。首先,如果你正在使用 GitHub,请安装 Hub 作为 git 的包装器;它扩展了命令以包含诸如 forks、pull requests 和 pages 等概念。其次,设置你的提示符(有很多项目可以在各种 shells 中工作;可以自己找找或者在 Twitter 上问我 以获得最新的建议)来反映你当前的 git 状态——你所在的哪个分支,是否有未提交的工作等等。细微的视觉提示将帮助你避免犯可怕的错误。