摘要:本文聚焦「Git 极客生存指南:从命令行到私有云搭建」,梳理核心概念、关键方法与落地实践。
面向对象: 程序员 (Programmers)。 目标: 掌握 Git 的“黑魔法”,处理复杂的分支管理、冲突解决,并学会搭建团队的私有代码仓库。
1. 常用命令速查 (The Cheat Sheet)
1.1 基础操作
1.2 分支操作 (Branching)
1.3 后悔药 (Undo)
1.4 暂存现场 (Stash)
当你正在修 Bug,突然老板让你切分支去改另一个紧急 Bug:2. 进阶技巧 (Advanced Skills)
2.1 Rebase (变基) vs Merge
- Merge: 保留真实的历史记录,会有 “Merge branch ‘xxx’” 的提交。适合公共分支合并。
- Rebase: 把你的提交“接”在目标分支的最新提交后面。历史记录是一条直线,非常干净。
git pull --rebase: 拉取代码时自动变基 (推荐配置)。git rebase develop: 在 feature 分支上,把 develop 的最新代码垫在下面。
2.2 Cherry-pick (摘樱桃)
只想要某个分支里的某一次提交,而不是整个分支?2.3 解决冲突 (Conflict Resolution)
A. 文本文件冲突
-
定位: 打开冲突文件,找到
<<<<<<<,=======,>>>>>>>标记。 - 修改: 决定保留哪部分代码(或者都保留),删除标记符号。
-
提交:
git add+git commit。
B. 二进制文件冲突 (Binary Conflict) - 关键!
图片、模型、DLL 无法合并内容,必须二选一。 命令行方案:-
保留我的 (Mine): 我改了图,我要覆盖服务器的。
-
保留他的 (Theirs): 别人的图是对的,我放弃我的修改。
-
注意:在
git merge时,--ours是指当前分支,--theirs是指要合并进来的分支。但在git rebase时,逻辑是反的!务必先备份。
- 在冲突文件上右键。
-
选择
Resolve using 'Mine'(使用我的版本) 或Resolve using 'Theirs'(使用远程版本)。 - 工具会自动执行上述命令并标记为已解决。
- 使用 LFS 的锁定功能:
git lfs lock image.png。 - 这样当你在修改时,别人无法推送这个文件,直到你
unlock。
3. 搭建私有 Git 服务器 (Self-Hosted Git)
对于不想把代码放在 GitHub/Gitee 的团队,推荐搭建 Gitea (轻量级) 或 GitLab (功能全)。3.1 方案 A: Gitea (推荐,极轻量)
适合小团队,一个二进制文件搞定,内存占用极低。 搭建步骤 (Windows/Linux):- 下载: 去 Gitea 官网 下载对应系统的可执行文件。
- 运行: 直接双击运行 (会启动 Web 服务器,默认端口 3000)。
-
配置: 浏览器访问
localhost:3000,首次运行会进入安装向导。- 数据库选
SQLite3(最简单,无需安装 MySQL)。 - 设置管理员账号。
- 数据库选
-
局域网访问: 确保防火墙开放 3000 端口。队友可以通过
http://192.168.x.x:3000访问。
3.2 方案 B: 局域网裸仓库 (Bare Repo)
最原始的方法,不需要任何 Web 界面。-
服务器端 (找台电脑做主机):
-
共享: 将
vampirefall.git文件夹设置为网络共享文件夹 (Windows SMB)。 -
客户端:
4. Unity 项目的 .gitignore (必抄)
5. LFS 配置 (大文件存储)
对于大于 100MB 的文件 (PSD, FBX),必须用 LFS。-
安装:
git lfs install -
配置:
-
提交: 这会生成一个
.gitattributes文件,务必把它提交上去。
一句话忠告: 永远不要在主分支 (master/develop) 上直接写代码。 Commit 早,Commit 勤。
🐙 Git 版本管理与 Commit Log 规范
核心理念: Commit Log 是写给人看的,不是写给机器看的。 一个好的 Commit Log 应该能回答三个问题:
- 改了什么? (What)
- 为什么改? (Why)
- 怎么改的? (How - 可选,如果是复杂逻辑)
1. Commit Message 格式规范
采用业界标准的 Angular Commit Convention,结构如下:1.1 Type (必填)
用一个词描述改动的性质:- feat: 新功能 (Feature)。
- fix: 修补 Bug。
- docs: 仅修改了文档 (如 README)。
- style: 格式修改 (不影响代码运行的变动,如空格、缩进)。
- refactor: 重构 (即不是新增功能,也不是修改 bug 的代码变动)。
- perf: 性能优化。
- test: 增加测试或修改测试。
- chore: 构建过程或辅助工具的变动 (如 .gitignore, package.json)。
- art: 美术资源提交 (贴图、模型、预制体)。
1.2 Scope (选填)
用括号说明影响的范围 (模块/功能):feat(Tower): 塔防模块。fix(UI): 界面模块。art(VFX): 特效资源。
1.3 Subject (必填)
简短的描述,不超过 50 个字符。- 原则: 动词开头,使用祈使句。
- Good: “Add double jump mechanic” (添加二段跳机制)
- Bad: “Fixed some bugs” (修了一些bug -> 修了啥??)
1.4 Body (选填,但推荐)
详细描述。- 解释为什么要做这个修改?
- 解释之前是怎么样的,现在是怎么样的?
- 如果是修复 Bug,描述复现步骤或根因。
1.5 Footer (选填)
- 关联的 Issue 或任务 ID。
Closes #123BREAKING CHANGE: 如果有破坏性更新(如改了存档格式),必须大写注明!
2. 📝 标准 Commit Log Demo (抄作业区)
请团队成员直接复制以下模板修改。场景 A: 修复了一个 Bug
场景 B: 开发了一个新功能
场景 C: 提交美术资源
场景 D: 性能优化
3. 分支管理策略 (Branching Strategy)
3.1 分支命名
- master / main: 随时可发布的稳定版本。绝对禁止直接 Push。
- develop: 开发主分支。所有 Feature 分支合入这里。
- feat/xxx: 功能分支。如
feat/login_system。 - fix/xxx: 修复分支。如
fix/crash_on_start。 - art/xxx: 美术资源分支。
3.2 工作流 (Workflow)
- 接到任务 “开发登录系统”。
-
基于
develop切出feat/login。 - 开发… 提交… (多次 Commit)。
- 开发完毕,推送到远程。
-
发起 Pull Request (PR) 合入
develop。 - Code Review: 同事检查代码,确认无误后 Approve。
- 合并。
4. 工具强制约束 (Enforcement)
为了防止人为偷懒,建议部署 Git Hooks。4.1 commit-msg Hook
在.git/hooks/commit-msg 中添加脚本,使用正则表达式检查 Commit Message 格式。如果不符合 <type>(<scope>): <subject> 格式,直接拒绝提交。
4.2 pre-commit Hook
在提交前自动运行:- 代码格式化 (CSharpier / Format)。
- 简单的静态检查 (如有无带
Debug.Log的代码)。
最后通牒: “Update”, “Fix bug”, “Backup”, ”…” 这种 Commit Message 一经发现,请请全组喝奶茶。
📚 扩展阅读与参考标准 (References)
🌍 行业标准
- Conventional Commits
- 本文档基于此规范。它是目前最流行的 Commit Message 标准,被 Angular, React, Electron 等数万个开源项目采用。
- Semantic Versioning 2.0.0 (语义化版本控制)
- 解释了为什么
BREAKING CHANGE会导致大版本号 +1 (v1.0.0 -> v2.0.0)。
- 解释了为什么
🔧 自动化工具
- Husky
- 最流行的 Git Hooks 工具。可以用它在
git commit之前自动运行 Lint 检查。
- 最流行的 Git Hooks 工具。可以用它在
- Commitlint
- 一个命令行工具,用来检查 Commit Message 是否符合 Conventional Commits 规范。建议集成到 CI/CD 流程中。
📖 深度文章
- How to Write a Git Commit Message (Chris Beams)
- 这篇博客被无数人引用,详细解释了“为什么要用祈使句”、“为什么首行不能超过50个字符”。
🐙 GitHub 工作流与 PR 最佳实践
核心理念: 主分支 (main/develop) 是神圣不可侵犯的。任何代码想要进入主分支,必须经过至少一双眼睛的检查 (Code Review)。这个过程就叫 Pull Request (PR)。
1. 标准工作流 (The Flow)
1.1 Fork vs Branch
- 开源模式 (Fork): 你没有原仓库的写权限。你把仓库
Fork到你自己名下,改完后向原仓库发起 PR。 - 团队模式 (Branch): 你有写权限。你直接在原仓库里切一个
feature/xxx分支,改完后向develop发起 PR。 - Vampirefall 推荐: 团队模式。效率更高。
1.2 完整生命周期
-
新建分支: 基于最新
develop创建feat/tower_fire。 -
提交代码: 在
feat/tower_fire上 commit。 -
发起 PR: 在 GitHub/Gitea 网页上点击 “New Pull Request”。
- Source:
feat/tower_fire - Target:
develop
- Source:
- Code Review: 你的同事收到通知,进来检查代码,写评论。
-
修改反馈: 根据同事的建议,继续在
feat/tower_fire上提交修改。 - 合并 (Merge): 同事点赞 (Approve) 后,点击 “Squash and Merge”。
-
删除分支: 完事后删掉
feat/tower_fire。
2. 如何写一个优秀的 PR 描述?
PR 的描述决定了 Reviewer 的心情和审核速度。2.1 标题 (Title)
- 格式:
<Type>: <Subject>(同 Commit Message) - 例子:
feat: 实现火焰塔的燃烧逻辑
2.2 模板 (Template)
建议在仓库根目录建一个.github/PULL_REQUEST_TEMPLATE.md,内容如下:
3. Code Review 礼仪与标准
3.1 Reviewer (审核者) 的职责
- 看逻辑: 代码是否实现了需求?有没有明显的 Bug?
- 看规范: 变量名是否规范?有没有写注释?
- 看性能: 有没有在 Update 里
new List?有没有死循环风险? - 语气: 对事不对人。
- Bad: “你这代码写得太烂了。”
- Good: “这里可能会产生 GC,建议改用对象池。“
3.2 Submitter (提交者) 的心态
- 不要玻璃心: 别人指出的问题是为了项目好,不是针对你。
- 解释: 如果你不认同 Reviewer 的意见,请在评论里解释你的理由,或者线下沟通。
- 及时响应: 别发了 PR 就不管了,别人提了意见赶紧改。
4. Merge 策略:Squash vs Merge
点击 Merge 按钮时,有三种选项:4.1 Create a merge commit (普通合并)
- 保留所有历史记录。如果你的分支上有 100 个 “fix typo” 的垃圾提交,它们都会进入主分支。
- 评价: ❌ 脏。不推荐。
4.2 Squash and merge (压缩合并) - 推荐
- 把你分支上的 100 个提交压缩成 1 个提交,合入主分支。
- 评价: ✅ 干净。主分支的历史记录非常清晰,一个功能对应一个 Commit。
4.3 Rebase and merge (变基合并)
- 把你的提交直接接到主分支后面,像从来没分叉过一样。
- 评价: ⚠️ 高风险。如果不仅保留了垃圾提交,还没有 Merge 节点,出问题很难回退。
5. 常见问题
- PR 冲突了怎么办?:
- 在本地
git pull origin develop(把主分支最新代码拉下来)。 - 本地解冲突。
git push更新你的 PR 分支。GitHub 会自动更新状态。
- 在本地
- PR 太大了怎么办?:
- 如果一个 PR 改了 50 个文件,没人愿意看。
- 拆分: 先提一个
feat/tower_base(只有基类),合入后再提feat/tower_fire。
一句话总结: PR 是代码质量的守门员。 没有 Review 的代码,就是埋在项目里的雷。
🐢 SVN vs 🐙 Git:深度对比与极简上手指南
写在前面: 很多团队(尤其是美术同学)习惯了 SVN 的“直观”,对 Git 感到恐惧。其实 Git 并没有那么难,只是逻辑变了。 核心区别: SVN 是集中式的 (服务器坏了大家都得停工),Git 是分布式的 (每个人电脑里都有一份完整的版本库)。
1. 深度对比分析 (Analysis)
1.1 为什么美术喜欢 SVN?
- 文件锁定 (Locking): SVN 可以在你编辑
Hero.psd时“锁住”它,别人就不能改了。这对二进制文件(图片、模型)至关重要,防止冲突。Git 默认没有锁(需要 LFS 插件)。 - 局部检出: SVN 可以只下载项目的
Art/Characters文件夹。Git 必须把整个仓库(包括你不关心的代码和文档)全拉下来。 - 权限控制: SVN 可以精确控制“美术只能读写 Art 目录,不能碰 Code 目录”。Git 通常是仓库级别的权限(要么都能读写,要么都不能)。
1.2 为什么程序喜欢 Git?
- 分支 (Branching): Git 切分支是秒级的。每个功能一个分支,互不影响。SVN 切分支很重,而且合并代码简直是噩梦。
- 离线工作: Git 可以在没网的时候提交代码(Commit 到本地),等有网了再推送到服务器。SVN 没网就废了。
- 安全性: Git 每个人的电脑里都有完整备份。服务器炸了,随便找台电脑就能恢复。SVN 服务器炸了且没备份,项目就没了。
1.3 结论:Vampirefall 该选谁?
- 代码/配置: 必须用 Git。分支管理和合并是多人协作的刚需。
- 美术大资源 (PSD/MAX):
- 方案 A: 继续用 SVN 管理源文件,导出后的资源 (FBX/PNG) 进 Git。
- 方案 B: 全面转 Git,但必须开启 Git LFS (Large File Storage) 并配置文件锁。
2. 🐙 Git 极简上手指南 (美术/策划专用版)
忘掉命令行! 我们推荐使用 Sourcetree 或 GitHub Desktop 或 TortoiseGit (小乌龟,长得跟 SVN 很像)。2.1 核心概念对应表 (SVN -> Git)
| 你在 SVN 做的操作 | 在 Git 里的对应操作 | 区别 |
|---|---|---|
| Update (更新) | Pull (拉取) | 没区别,都是把服务器的东西拉下来。 |
| Commit (提交) | Commit (提交) + Push (推送) | 这是最大的坑! SVN 提交就完事了。 Git 提交只是存到你自己电脑里,必须再点一下 Push 才能传到服务器给别人看。 |
| Revert (还原) | Discard / Reset | 放弃修改,还原到上次提交的状态。 |
| Lock (锁定) | LFS Lock | 需要专门配置 LFS 才能用。 |
2.2 傻瓜式工作流 (Daily Workflow)
假设你使用 TortoiseGit (因为它和 SVN 操作最像):第一步:早上开工 (Pull)
-
在项目文件夹上右键 ->
TortoiseGit->Pull。 -
点
OK。 - 目的: 确保你拿到的是最新版本,防止和别人冲突。
第二步:干活 (Work)
- 改 Excel,画图,改场景… 随便弄。
第三步:下班提交 (Commit + Push)
-
右键 ->
Git Commit -> "master"。 - 勾选你修改的文件。
-
在 Message 框里写:
art: 修改了吸血鬼主角的模型贴图。 -
点击
Commit。 -
关键动作: 此时弹窗左下角会有一个
Push按钮,一定要点一下! (或者 Commit 完单独右键 ->Push)。 - 目的: 只有 Push 成功了,你的东西才算真正提交了。
2.3 遇到冲突怎么办? (Conflict)
- 现象: Push 失败,提示
Updates were rejected。 - 原因: 你改了
Data.xlsx,小王也改了Data.xlsx,而且他比你先 Push。 - 解决:
-
先点
Pull。Git 会试图合并。 - 如果合并失败,文件上会有个感叹号。
- 策划/美术: 别慌!直接找程序帮忙,或者备份你的文件,还原 (Revert),拉取最新 (Pull),再把你的改动覆盖上去。
- 程序: 使用 Merge Tool 解决冲突。
-
先点
3. 🚀 给主程的建议:如何平滑迁移?
- 保留 SVN 习惯: 给美术装 TortoiseGit,因为右键菜单的操作习惯和 SVN 几乎一样,学习成本最低。
-
忽略文件配置 (.gitignore):
- 务必把
Library/,Temp/,Logs/,.vs/屏蔽掉。SVN 以前可能把这些垃圾都传上去了,Git 绝对不行。
- 务必把
-
LFS 强制开启:
- 配置
.gitattributes,把*.psd,*.fbx,*.png,*.wav全部走 LFS。否则 1个月后你的 Git 仓库会大到拉不下来。
- 配置
一句话总结: Git = SVN + “本地仓库”。 以前是
写完 -> 上传。
现在是 写完 -> 存本地 (Commit) -> 上传 (Push)。
多了一步,但更安全。