Git¶
配置¶
设置 Commit 时的用户名:
git config --global user.name "[name]"
设置 Commit 时的邮箱:
git config --global user.email "[email address]"
启用有帮助的彩色命令行输出:
git config --global color.ui auto
分支¶
创建¶
用 commit hash 创建分支:
git branch branch_name <commit-hash>
用符号引用(symbolic reference)创建分支:
git branch branch_name HEAD~3
创建分支并切换:
git checkout -b branch_name <commit-hash or HEAD~3>
推送¶
将分支推送到远程:
git push origin local_branch_name:remote_branch_name
本地关联远程分支(这样 push 或 pull 时不需要再指定分支):
git branch --set-upstream-to=origin/remote_branch_name local_branch_name
将本地分支推送到远程,同时本地关联远程分支(本地和远程的分支都叫 branch_name):
git push -u origin branch_name
删除¶
删除本地分支前,先切到其他分支。
删除本地分支,如果分支没被合并则不允许删除:
git branch -d branch_name
强制删除本地分支,不管是否被合并:
git branch -D branch_name
删除远程分支:
git push origin -d branch_name
推送空分支到远程,也能删除远程分支:
git push origin :branch_name
删除远程分支后,更新本地分支列表(-p
即 --prune
):
git fetch -p
重命名¶
重命名本地分支:
git branch -m old_name new_name
HEAD 引用¶
HEAD 文件通常是一个符号引用(symbolic reference),指向目前所在的分支。 所谓符号引用,表示它是一个指向其他引用的指针。
- HEAD 指向当前所在分支提交至仓库的最新一次的 commit。
-
~
是用来在当前提交路径上回溯的修饰符。HEAD~{n} 表示当前所在的提交路径上的前 n 个提交(n >= 0):
HEAD = HEAD~0
HEAD~ = HEAD~1
HEAD~~ = HEAD~2
^
是用来切换父级提交路径的修饰符。
当我们始终在一个分支比如 dev 开发/提交代码时,每个 commit 都只会有一个父级提交,就是上一次提交,但当并行多个分支开发,feat1, feat2, feat3,完成后 merge feat1 feat2 feat3 回 dev 分支后,此次的 merge commit 就会有多个父级提交。
# 当前提交 HEAD = HEAD~0 = HEAD^0 # 主线回溯 HEAD~1 = HEAD^ # 主线的上一次提交 HEAD~2 = HEAD^^ # 主线的上二次提交 HEAD~3 = HEAD^^^ # 主线的上三次提交 # 如果某个节点有其他分支并入 HEAD^1 # 主线提交(第一个父提交) HEAD^2 # 切换到了第2个并入的分支并得到最近一次的提交 HEAD^2~3 # 切换到了第2个并入的分支并得到最近第 4 次的提交 HEAD^3~2 # 切换到了第3个并入的分支并得到最近第 3 次的提交 # ^{n} 和 ^ 重复 n 次的区别 HEAD~1 = HEAD^ HEAD~2 = HEAD^^ HEAD~3 = HEAD^^^ # 切换父级 HEAD^1~3 = HEAD~4 HEAD^2~3 = HEAD^2^^^ HEAD^3~3 = HEAD^3^^^
Windows CMD 中的坑¶
在 Windows CMD 中,^
是转义符。输入单个 ^
的话,命令行就会在下一行问 More?
让你继续输入需要转义的内容。^^
才会被识别为 ^
。
要想让 git 正确识别 HEAD^
,需要输入 HEAD^^
或者用双引号包裹 "HEAD^"
。换 powershell、git bash 也行。
路径大小写不敏感¶
Git 默认不区分路径大小写,类似 Windows。比如把 Test
改名为 test
是没用的,Git 认为没有发生变化。
可以设置
git config core.ignorecase false
让 Git 区分大小写,但是不推荐,我遇到过问题:执行这条命令后,把 Test
改名为 test
再提交,远程仓库里同时存在 Test
和 test
,但是本地仓库里只有 test
。
上面的问题只能用最朴素的办法来解决:
- 把
Test
改名为Test1
- 提交
- 把
Test1
改名为test
- 提交
Sync Fork¶
直接在 GitHub 上点 Sync Fork
有可能产生一个新的 Commit,污染提交记录。下面用 Rebase 来同步,保证干净的提交记录。
-
添加原来的远程库,可以起名为
fork
:git remote add fork https://github.com/com/original/original.git
-
拉取原来的远程库的信息:
git fetch fork
-
切到需要同步的本地分支,然后 rebase:
git checkout branch_name git rebase fork/main
-
强制推送至云端:
git push origin +branch_name
+
是强制推送的意思,也可以用下面的方法:git push -f origin branch_name
Squash Merge¶
将多个提交合并成一个,然后 Merge,需要自己再写一个 Commit:
git merge --squash another_branch
git commit -m "message here"
删除中间几个 Commit¶
使用交互式 Rebase 来删除中间的 Commit:
git rebase -i <commit-hash>
这行命令会先在编辑器中打开一个文件,在文件中可以给不想要的 Commit 标记上 drop,然后保存关闭文件。之后 git 会执行 Rebase,删掉标记了 drop 的 Commit。
Pull¶
git pull
是 git fetch
+ git merge
。
撤销¶
常用工作流¶
LFS¶
Git Large File Storage (LFS) 是 Git 的一个扩展。它可以将较大的二进制文件替换为较小的文本文件,然后把真实文件的内容保存在其他地方。
-
下载 并安装 Git LFS。然后,运行下面的命令来为账号初始化:
git lfs install
每个账号只需要运行一次。
-
在 Git 仓库中运行下面的命令(或者直接编辑 .gitattributes),添加需要 Git LFS 管理的文件类型:
git lfs track "*.psd"
接下来,要保证 .gitattributes 被追踪:
git add .gitattributes
在任何时候都可以运行上面的命令,指定新的文件类型。然而,这个命令不会把之前匹配的文件转换到 Git LFS 中,比如其他分支里的文件,或者之前提交历史里的文件。
如果还要转换之前的文件,需要使用
git lfs migrate
命令,并指定好参数。
现在,只要像以前一样使用 Git 就行了!例如,当前的分支是 main
:
git add file.psd
git commit -m "Add design file"
git push origin main
相关文章