Githug
他喵的这是个啥!?难道不是 GitHub 拼错了么,和 Git 什么关系? 和游戏又有什么关系?
其实,他的元身在这里:https://github.com/Gazler/githug ,这个命令行工具被设计来练习你的 Git 技能,它把平常可能遇到的一些场景都实例化,变成一个一个的关卡,一共有 55 个关卡,所以将他形象的形容为 Git 游戏。
既然是游戏,作为一个专业的游戏玩家,通关自然是我的最终目标了!!!
安装游戏
没什么好说的,终端运行如下命令即可,如果碰到了墙,自行搬梯子
data:image/s3,"s3://crabby-images/913b7/913b7df64eeeecca8ab9ddee7798a551c111141d" alt=""
游戏开始
直接输入 githug
就可以开始游戏了!开始的时候会询问是否创建文件夹,输入 y
确认创建,以后的操作将都在这个文件夹中进行。
data:image/s3,"s3://crabby-images/ddcd4/ddcd44936ade3557cbad5f51567843b9232d24aa" alt=""
第一关(Init)
紧接着,马上进入到了第一个关卡, 按照提示初始化这个这个 githug
文件夹为仓库。完成关卡可以通过调用 githug play
验证操作,成功则会进入下一个关卡
data:image/s3,"s3://crabby-images/82647/826472c97259bbb27288dca7e7a0cdc79197c599" alt=""
第二关(Config)
设置 Git 用户名和邮箱,为了不影响全局的配置,我设置的是仓库级别的。
data:image/s3,"s3://crabby-images/ac624/ac6241d48a6f4603ff1d38bb011ce6fca091c9dd" alt=""
第三关 (Add)
使用 add
命令将 README 文件添加到 staging area.
data:image/s3,"s3://crabby-images/5fd98/5fd98cf96020537315bbb1acac017b1e8dfbcdb8" alt=""
第四关 (Commit)
提交 README 文件,记得每次 commit
使用 -m
参数加上备注是个好习惯
data:image/s3,"s3://crabby-images/fba0a/fba0a93710790c9b149f843d7e418813e07e7994" alt=""
第五关(Clone)
克隆一个仓库,默认的文件夹名是远端的仓库名
data:image/s3,"s3://crabby-images/bd717/bd717863835eaf3c97c89be7fd3e6f5dc0583f08" alt=""
第六关(Clone to folder)
同样是克隆一个仓库,不同的是可以指定一个文件夹名
data:image/s3,"s3://crabby-images/0b0aa/0b0aa88708fb8d853bbed6c8155acb2e43ab0fc9" alt=""
第七关(Ignore)
忽略所有 .swp
后缀名的文件。这里使用 vim 编辑器打开 .gitignore
,这个文件记录了 git 忽略文件的规则, 不会 vim 的同学可以用自己熟悉的编辑器。
data:image/s3,"s3://crabby-images/bc599/bc59966e348556544c5a1d09ccc6492e912074d6" alt=""
使用正则(glob 模式)匹配所有的 .swp
文件,然后保存并退出
data:image/s3,"s3://crabby-images/66483/66483d8ed0cfb643c25f2fdf607789323d10fd1e" alt=""
第八关(Include)
除了 lib.a
文件,其他所有的 .a
后缀名的文件都忽略。和上一关的操作一样,修改 .gitignore
文件
data:image/s3,"s3://crabby-images/7184c/7184c37c0e628f7cc9812b7f08cd9b52aa3877cb" alt=""
其中 #
开头的是注释,用 *.a
匹配所有 .a
文件,!
开头代表不要忽略
data:image/s3,"s3://crabby-images/03668/0366848c6c2798f7ad609ef6f7a40e629186a97c" alt=""
第九关(Status)
查看所有处于 untracked
状态的文件。使用 git status
查看当前仓库的状态,可以看到红色部分就是 untracked
状态的文件
data:image/s3,"s3://crabby-images/3529e/3529eda3cd87f1e69aea60de5d0fabbb4273e685" alt=""
第十关(Number of files committed)
其实就是查看处于 staged 状态的文件,图中黄色部分就是,所以个数就是2
data:image/s3,"s3://crabby-images/72ed4/72ed451052ef5af586b89ea84ad641bf157f7a33" alt=""
第十一关(rm)
有一个文件从硬盘中删除了,但是并未从 git 仓库中删除,找到它并从 git 仓库中删除。删除也是修改的一种,提交这个修改就好了
data:image/s3,"s3://crabby-images/0e420/0e4201f7d673eadb47f5f1db53d00eab19cb6887" alt=""
第十二关(rm cached)
讲一个新文件从 staging area
中删除。按照要求,不应该直接从硬盘上删除这个文件,只是从 Git 中删除而已。加上 --cache
可以是文件只是从 staging area
中移除,不会真正的删除物理文件,如果要连这个物理文件也一起删除,请使用 -f
选项
data:image/s3,"s3://crabby-images/826a5/826a512bd84a28632ccfcb4b40821a1ab86bdd2b" alt=""
第十三关(stash)
临时提交某个文件。这个操作在需要临时保存修改,而又不想提交的时候特别好用!而且 git 中维护了一个栈来保存,所以支持提交多次。如果需要恢复某次提交,使用 git stash apply
即可。
data:image/s3,"s3://crabby-images/54392/543928ecec4acec3bbae1025ab18b642b275851b" alt=""
第十四关(Rename)
重命名文件。首先这个文件需要是已经是已追踪状态,才可以使用 git mv
命令,操作完成后自动处于 staging 状态
data:image/s3,"s3://crabby-images/fcfe3/fcfe34bcf30c3d60c5d9ae5f2c873340128686f6" alt=""
第十五关(Restructure)
移动所有 .html
文件到 src
文件夹。git mv
后面的第二个参数可以接受文件或目录,如果是目录,则文件会直接放入目录内,可以使用正则(glob模式)匹配所有 .html 文件
data:image/s3,"s3://crabby-images/ceb74/ceb74000dc6dc331660993ab084f7e59b5d8ad76" alt=""
第十六关(Log)
找到最新的 commit 的 hash 值。使用 git log
查看历史提交记录, 找到最新的 commit 的 hash 值,记录下来用户回答问题
data:image/s3,"s3://crabby-images/e9da3/e9da3f0db51474d30eaaa96507b4d5d4df072cda" alt=""
这里是按照倒叙排列的,最新的在最前面,commit
关键字后面跟着的就是这个 commit 的 hash 值
data:image/s3,"s3://crabby-images/5cb5e/5cb5e79afb8c596e1a2880f518ae13f669aab8e5" alt=""
第十七关(Tag)
为最新的 commit 打 tag。不加额外参数就是为当前 commit 记录 tag, 当然可以为特定的 commit 打
data:image/s3,"s3://crabby-images/02f7c/02f7c49bfc73a0d03d327b3dfb16fe5486957fbb" alt=""
第十八关(Push tags)
将所有本地 tag 都推送到远端。--tags
参数代表将所有的 tags 都推送到远端
data:image/s3,"s3://crabby-images/ee190/ee190a1316f908aa0d0ca17e35b57c738d5f5137" alt=""
第十九关(Commit amend)
某个文件在上次提交中遗漏了,在那次提交中补上这个文件。 其实,使用 git commit --amend
会进入编辑界面修改备注信息,我这里直接 :wq
保存并退出
data:image/s3,"s3://crabby-images/5cbe5/5cbe528530111ca2eaf00c1c578412e02a737574" alt=""
第二十关(Commit in feature)
为提交指定一个未来的时间。
data:image/s3,"s3://crabby-images/dc76a/dc76a9aeff17f5cc4c90cb344bf7627740421979" alt=""
data:image/s3,"s3://crabby-images/9eef7/9eef7e498d100e135158008d85bf60e47ee98c7e" alt=""
第二十一关(Reset)
两个文件都被添加到了 staging area
, 但是只想提交其中一个。使用 git reset
可以用仓库中的版本覆盖 staging area
的版本。
-
git reset
使用仓库中的版本覆盖staging area
中的,如果working directory
该文件没有其他修改,则staging area
中的修改将应用到working directory
中。反之working directory
中的版本将被保留,丢弃staging area
中的修改。 -
git checkout
则是使用staging area
的中的版本覆盖working directory
。
Paste_Image.png
第二十二关(Reset soft)
撤销上一次提交。
-
--soft
参数将上一次的修改放入staging area
-
--mixed
参数将上一次的修改放入working directory
-
--hard
参数直接将上一次的修改抛弃
第二十二关
第二十三关(Checkout file)
抛弃某一次的修改,使用上次提交的版本。checkout
和 reset
的区别参照第二十一关
data:image/s3,"s3://crabby-images/867f2/867f28130713fd56767df947efcf616bf53f200f" alt=""
第二十四关(Remote)
查看远端仓库。其实可以不加-v
参数,加这个参数只是可以将地址也一起输出(没想到下一关就是考察这个参数,平常习惯加这个参数了。。。)
data:image/s3,"s3://crabby-images/02488/0248861722d9bc4f96d90089c88fad05b61b8b30" alt=""
第二十五关(remote url)
查看远端仓库的 URL
data:image/s3,"s3://crabby-images/b8fd3/b8fd39bfece9098263ad7bd2b7f1e28eb6aee453" alt=""
第二十六关(pull)
拉取远端仓库。
其实可以指定分支,格式如下
git pull origin remote : local
对应的推送的格式如下
git push origin local : remote
需要注意的两个操作的分支顺序是相反的,记忆的方法很简单,拉取是从远端到本地,所以远端在前,而推送是从本地到远端,所以本地在前。
data:image/s3,"s3://crabby-images/4a122/4a1226631fe87ef3f89f5880df32fc786cbe8ed4" alt=""
第二十七关(Remote add)
添加一个远端仓库
data:image/s3,"s3://crabby-images/e3543/e35436517fede1cfb5601a43f0929d1eca057011" alt=""
第二十八关(Push)
推送本地修改到远端
data:image/s3,"s3://crabby-images/28d02/28d020f5043289edf1f491fc4952b58802e24411" alt=""
第二十九关(Diff)
查看 staging area
和 working directory
中文件的差异。
-
git diff
: 查看working directory
与staging area
之间的差异 -
git diff --cached
: 查看repository
与staging area
之间的差异 -
git diff HEAD
: 查看working directory
与repository
之间的差异
第二十九关
对应 git diff
的显示结果
data:image/s3,"s3://crabby-images/9ecc6/9ecc6e518119afcb048ff2814c5056bd64e6c5cd" alt=""
第三十关(Blame)
查看某个文件的修改人。这个命令简直邪恶,锅终于有人背了!!!
data:image/s3,"s3://crabby-images/68f76/68f766cdf42750cb9a44d9e231dc56cf063830fc" alt=""
git blame
列出了文件中每行的修改人是谁
data:image/s3,"s3://crabby-images/0c946/0c946e7731102a81298316405c57fe207d974e5c" alt=""
第三十一关(Branch)
创建一个分支
data:image/s3,"s3://crabby-images/27ccb/27ccbb6f0f26296fc27f9a78718ed31f4fcabda6" alt=""
第三十二关(Checkout)
创建一个分支,并切换过去。其实,git checkout -b my_branch
就是创建一个分支,并切换过去,而且这种方法更方便,平常用的更多
data:image/s3,"s3://crabby-images/151c9/151c9cc0aab14825bacb3f8733dcdb7cfba2d2bc" alt=""
第三十三关(Checkout tag)
切换到某个特定的 tag
data:image/s3,"s3://crabby-images/671f0/671f07b72fcaec7ed20f74c5e4b3af65a40752bf" alt=""
第三十四关(Checkout tag over branch)
切换到某个特定的分支,但是分支名和标签名重叠了
data:image/s3,"s3://crabby-images/9495b/9495ba673c74e0b59d81e64941291fdab9296353" alt=""
第三十五关(branch at)
根据一个特定的提交创建新分支
data:image/s3,"s3://crabby-images/a27f5/a27f56eee3cf203ebc4443aadc8a63b54c088590" alt=""
第三十六关(delete branch)
删除一个分支
data:image/s3,"s3://crabby-images/fe730/fe7304eac34f7bf8da66e01a947444f5067d94ce" alt=""
第三十七关(Push branch)
将分支推送到远端仓库
data:image/s3,"s3://crabby-images/c5b04/c5b0486934f5436fea00bbf7d7cac4463b1bb732" alt=""
第三十八关(merge)
合并分支。为了简化分支模型,可以使用 rebase
代替,后续关卡会遇到。
data:image/s3,"s3://crabby-images/c02c0/c02c063219ac8f791c784c659d22b07ee87fa0c8" alt=""
第三十九关(fetch)
获取远端的修改,但是并不合并到当前分支。其实,git pull
就是 git fetch
和 git merge
组成的。
data:image/s3,"s3://crabby-images/c8904/c8904acb57863d82c5e7be0408c53961c1d06ce3" alt=""
第四十关(rebase)
其实不知道怎么翻译 git rebase
这个命令。大概意思是从某个提交分化出两个分支,然后其中一个分支需要将另一个分支的修改合并过来,但是又不想在提交记录上留下两个分支合并的痕迹,只留下一个分支以前后顺序记录两边的修改。
git rebase
一个分支的所有修改在另一个分支上重新应用一遍,所以在提交记录上看,会发现一个分支的所有提交在另一个分支之前或者之后。然后删除另一个被合并的分支,保持分支简洁。
git rebase master feature
表示将 feature
上的修改在 master
上重新应用一遍
data:image/s3,"s3://crabby-images/1a6dd/1a6ddd757549933bc370e334b2c5eb586fd7e777" alt=""
对应第一个 git log --graph -all
,--graph
会用图形化将提交记录显示出来,而--all
会显示所有分支的提交记录
data:image/s3,"s3://crabby-images/c2e06/c2e0630754f73e7e15cd711a2434b29e736c47de" alt=""
对应第而二个 git log --graph -all
,可以发现只保留了一个分支,看起来简洁了很多。
data:image/s3,"s3://crabby-images/69a86/69a864e648cbd419310dab802734f31c729fc812" alt=""
在使用此命令的时候,需要非常注意的是,不要 rebase 哪些已经推送到公共库的更新,因为此操作是重新应用修改,所以公共库的更新可能已经被其他协作者所同步,如果再次 rebase 这些修改,将可能zh
第四十一关(repack)
将版本库未打包的松散对象打包
data:image/s3,"s3://crabby-images/01d45/01d45bd2690e2635e64e8d7afd246d4f8e216678" alt=""
第四十二关(cherry pick)
应用某一个提交的修改。
data:image/s3,"s3://crabby-images/625ab/625ab6db07ea9849fa24ac43a10c255b32aa7788" alt=""
找到我们想要的那个提交,记录下它的 hash 值
ca32a6dac7b6f97975edbe19a4296c2ee7682f68
data:image/s3,"s3://crabby-images/4fd65/4fd65d1a80902bbc9a77fde08076ef5b692d4a83" alt=""
第四十三关 (Grep)
git grep
支持各种条件搜索及正则表达式,平时用的不多,但感觉功能强大。
data:image/s3,"s3://crabby-images/f1a78/f1a7895d000e782f7c34e5a3360009b769677a6e" alt=""
对应 git grep TODO
的结果
data:image/s3,"s3://crabby-images/83bea/83beab884a9b4e1c1c7ed61bfcec0a5db80d111d" alt=""
第四十四关(rename commit)
重命名提交。当涉及提交修改时,应该想到 git rebase -i
命令,它接受可以一个参数(提交的哈希值),它将罗列出此提交之后的所有提交,然后可以对个个提交做对应的操作。
data:image/s3,"s3://crabby-images/93c58/93c58602a9b3205fe933d5688475756e82c021f9" alt=""
重命名前的提交记录
data:image/s3,"s3://crabby-images/bb6c5/bb6c51586383f61bdccb8b7190689093e9e4e493" alt=""
将需要重命名的提交前的 pick
修改为 reword
data:image/s3,"s3://crabby-images/dfe08/dfe08db1b39c63544d452e36a1aeb27812a33f78" alt=""
修改成新的备注,保存并推出
data:image/s3,"s3://crabby-images/87776/87776402cd2a4ce638248a736f08dc0299f303b0" alt=""
第四十五关(squash)
合并多个提交。
data:image/s3,"s3://crabby-images/74065/74065aacee0b0d76f42df7447d31198fc533b271" alt=""
合并前的提交记录。
data:image/s3,"s3://crabby-images/1f995/1f9951c4725b062101c413a57fd464b889b1ca50" alt=""
将需要合并的提交前的 pick
改成 squash
或 s
。squash
代表并入前一个提交,保存并退出。
data:image/s3,"s3://crabby-images/3a997/3a997d7bb3e70d374008efd6fabc0ce3e12b33d2" alt=""
为新的提交修改备注
data:image/s3,"s3://crabby-images/e2154/e2154cb6ebe9bee2160265cc283980688fcfe1ec" alt=""
第四十六关(merge squash)
将某个分支上的所有修改都应用成一个提交。默认修改都将进入暂存区
data:image/s3,"s3://crabby-images/3be31/3be31c4e6ddf43e9e1d4c69fab486065c77d8c1e" alt=""
第四十七关
重新排列提交顺序。
data:image/s3,"s3://crabby-images/979d7/979d71fe8ec5a6d75fc6d942821a4842631d1c5c" alt=""
排序前,对应第一个 git log
data:image/s3,"s3://crabby-images/6471b/6471b849ffebe35bf0b5077f6d01de110f75b02a" alt=""
在编辑界面,将 Second commit
和 Third commit
的顺序调换
data:image/s3,"s3://crabby-images/66ab8/66ab87b907a789b9f97991910e4b79b484146f19" alt=""
排序后,对应第二个 git log
data:image/s3,"s3://crabby-images/fa5c5/fa5c535240355f0df923a0b6bab36884c53e31f9" alt=""
第四十八关
使用 git log
查看所有的提交记录,太长我就不全贴出来了,找到最开始的提交 f608824888b83bbedc1f658be7496ffea467a8fb
data:image/s3,"s3://crabby-images/415f3/415f3f5d8f0855ca47bbf5cfb771ef8fb5a9920e" alt=""
git bisect start master f608824888b83
中,master
是有 bug 的节点,f608824888b83
是没有 bug 的节点。
data:image/s3,"s3://crabby-images/9b6f7/9b6f7493c2a1a6bf794ebd358b1377991a475034" alt=""
第四十九关(Stage lines)
其实,提交文件的部分修改这种需求平时还是比较常见的,不过平时都是用 Source Tree 来操作的,但是看到这题之后,好像又开启了一扇大门。
data:image/s3,"s3://crabby-images/767cb/767cb7e1aa38c50fcda60124bc3b4b7c4e5227b6" alt=""
data:image/s3,"s3://crabby-images/0559e/0559e17c1c58f3a4f13abd97596605ef65c36128" alt=""
data:image/s3,"s3://crabby-images/4305f/4305fc693914a2c75d1195d929176d46c678d032" alt=""
第五十关(Find old branch)
git reflog
可以列出所有的操作记录,所以找到之前忘记的信息并不是什么难事
data:image/s3,"s3://crabby-images/f4f32/f4f32b1ed4e5dd739aae1bdec38cc6402c6f0b69" alt=""
对应 git reflog
的显示内容
data:image/s3,"s3://crabby-images/42e7e/42e7ec0ff1b2b937d9e1f6df6c86a7d4ffbf312e" alt=""
第五十一关(Revert)
与 reset
不同的是,revert 只会撤销当前的 commit,而之后的 commit 操作的修改还会保留,但是reset
还会将之后的所有 commit 操作的修改全部退回 staging area 或丢弃。
data:image/s3,"s3://crabby-images/9a2ca/9a2ca339c9dcfdc97178cff88f79f94b83369c47" alt=""
这是执行撤销操作前的记录,对应第一个git log
data:image/s3,"s3://crabby-images/743e7/743e736814f1e17b5d570c8d5ea77b89358bd73e" alt=""
撤销操作会生成一个新的 commit,保存并退出即可
data:image/s3,"s3://crabby-images/a1176/a11764dae2c516050ff1ed43b5c64693bdd841ba" alt=""
撤销之后的记录,它不破坏原有的记录,对应第二个 git log
data:image/s3,"s3://crabby-images/fb437/fb437f92322fdc48713a862b3cd7b6f011be699f" alt=""
第五十二关(Restore)
根据之前的经验,git reflog
可以查看所有的操作记录,所以只要能找到误操作之前的 commit id,一样能够恢复现场。
data:image/s3,"s3://crabby-images/6fcc6/6fcc696ab4760fc9431be8da0f3458c72cde8ec4" alt=""
执行 git reflog
后画面如下,根据操作记录,找到你误操作的之前的 commit id
data:image/s3,"s3://crabby-images/64aaa/64aaa6c6938da878cb162dfe4bc0e31dc1e3b0b0" alt=""
第五十三关(Conflict)
冲突处理在平常的协同工作中真是再常见不过了,需要注意的是存在冲突的文件是在 working directory 中的,在解决完冲突之后需要添加到 staging area 并提交。
data:image/s3,"s3://crabby-images/7619e/7619e24c6133efca8b987dc1502d59ce8f5f55b3" alt=""
其实冲突解决完成的图片丢失了,只能口述了。
<<<<<<< HEAD
到 =======
之间的内容代表 master
分支的修改,=======
到 >>>>>> mybranch
之间的内容代表 mybranch
分支的修改,保留 mybranch
分支的修改,删除master
分支的修改即可,当然这些特殊符号所在行也要一并删除。
data:image/s3,"s3://crabby-images/e0bde/e0bde034d20ce46e5d2309d080a39763ac9eb352" alt=""
第五十四关(Submodule)
submodule 是一个很方便的将一个仓库分解成多个子模块的命令,特别是项目比较大且依赖其他 Git 项目的时候,比如 Cocos2d-x。虽然好用,但是门槛也相对高点,如果维护好 submodule 还是需要好好研究一下。
data:image/s3,"s3://crabby-images/27d9a/27d9a62634f75e955913f882b196b4448b8b8c19" alt=""
第五十五关(Contribute)
其实到这里已经可以算是通关,如果感兴趣的话可以到 GitHub 为这个项目贡献代码。
data:image/s3,"s3://crabby-images/6b573/6b573aef39fda2cd881c1b23ebe2a4782377b000" alt=""
结尾
其实这里的所有关卡展示的内容只是 Git 的冰山一角,Git 的魅力远不止这些,还需我们慢慢探索~
如果文章有表述错误,欢迎指正。
最后,感谢这篇文章带我入坑:
http://gold.xitu.io/entry/5684844560b2cd25b7cb41a3
网友评论
41题: git rebase --onto master wrong_branch readme-update
另外,rename commit和git squash等几个题目的commit-id感觉图文不符。。应该是选择“合并点之前一次commit的commit id”
看了说明, 发现自己又学了一些新东东。