這是一個系列文章,介紹學習 Git 的一個小遊戲 - githug,若是你是第一次看到,請先閱讀:
闖過這 54 關,點亮你的 Git 技能樹
闖過這 54 關,點亮你的 Git 技能樹(一)
闖過這 54 關,點亮你的 Git 技能樹(二)
闖過這 54 關,點亮你的 Git 技能樹(三)
闖過這 54 關,點亮你的 Git 技能樹(四)git
沒想到第四彈在「開發者頭條」上得到了 300 多個贊。
看來你們對這個小遊戲挺感興趣的,並且還有很多朋友在促更,今天就讓咱們一氣呵成打通最後的 14 關吧。程序員
一樣,如對任何命令使用有疑問請看第一篇裏的推薦教程,也歡迎在下面留言,我會盡力提供幫助的。工具
項目時間長了,git
倉庫會慢慢變大,如何優化?
這個場景平常不多用到,並且不是必須用的,因此沒什麼概念。來看一下提示吧!
打開幫助以後輸入 /redundant
搜索關鍵字,一下就找到了一個 -d
的參數。
post
有一次,我正在一個特性分支上開發一個功能,提交了幾回代碼,就在準備結束合併代碼的時候,而後產品經理說這個需求不用作了。
我強壓下內心奔騰的一萬隻草泥馬,準備刪除這個分支,但在刪除以前想到有一個 commit 是對一個工具類的修改,仍是有用的。
這是我就須要從特性分支上把這個 commit 摘出來,合到 master 分支上,再刪除特性分支。
這個題目就是相似的場景,先來看看特性分支叫什麼,而後找到須要「摘」出來的那個 commit:
查看了一下 log
命令的幫助,發現能夠指定分支,這樣就省去了 checkout 到 new-feature 分支的步驟。
上圖中最後的一次提交就是咱們須要「摘」出來的,複製它的 Hash。
順利過關!學習
咱們在開發的過程當中,爲了避免影響當前正在作的事情,會把一些不那麼緊急的任務使用 TODO
註釋在代碼裏,現代的 IDE 都能幫咱們識別這些註釋並在一個單獨的窗口中羅列出來。
固然,不借助 IDE,光憑系統命令或 git 命令也是能夠作到的。
測試
查看 log,git log --oneline
,能夠看到中間的 commit message 有一個拼寫錯誤。
看一下提示,能夠在 rebase 的時候指定 -i
參數:
查看一下幫助就知道是 interactive
的意思:
在打開的 Vim 窗口中將第一行的 pick
改成 r
,表示:使用 commit,而且修改 commit message。
修正拼寫錯誤的 coommit
。
優化
當我使用 TDD 方式進行開發,會進行很是頻繁小步的提交,這樣在其餘同事看來就缺少完整性,也會增長後續維護成本。
因此 git 讓我能夠在 push 到遠程倉庫以前,對 commit 歷史進行修改合併,把多個 commit 合併成一個。
用 git log --oneline
看一下提交記錄:
從提交記錄能夠看出,它提示咱們將最後三個 commit 都合併到第二個 commit - Adding README 中。
接着執行 git rebase -i HEAD~4
:
第一個 commit 爲 pick
,後三個改成 s
,意思是使用這個 commit,但將它合併到前一個 commit 中去。
保存退出,會提示咱們編輯 commit message,再次保存退出後,查看一下提交記錄:
spa
題目要求在 merge 特性分支時,把全部的新提交合併成一個,先來看看 master 分支當成的狀態:git log --oneline
:
再看一下 git log long-feature-branch --oneline
:
翻譯
完整過關命令以下:
3d
最後 master 分支情況如圖,只用一個 commit,包含了 long-feature-branch
全部的修改:
提交順序錯亂時,也可使用 git rebase -i
進行調整。
先看看 Log,最後兩個提交顛倒了位置:
執行 git rebase -i HEAD~2
,將兩行 pick xxx 代碼交換位置便可。
看一下歷史:
代碼中不知道何時引入了 bug,不過不要緊,咱們有自動化測試。
咱們能夠不斷手工 checkout 到某個 commit,結合二分法查找快速定位到引入 bug 的那一個 commit。
不過這種純手工重複的事情,已經包含在 git 的命令中了,就是 bisect
讀一下 git help bisect
,能夠找到這個例子:
咱們知道 HEAD 的代碼是有問題的,而第一個 commit 的代碼是沒問題的。
經過 git log
得到第一個 commit 的 Hash,就能夠執行 bisect 命令:
紅線部分已經清楚地告訴咱們是哪一個 commit 引入的 bug 了。
有時開發了一個特性沒提交,接着又開發了另外一個特性。
做爲一個自律的程序員,應該是要分兩次提交的,若是修改的是不一樣的文件,那能夠輕鬆地經過 add 不一樣的文件進行兩次提交。
但此次好巧不巧的是竟然修改了同一個文件,怎麼辦?看看提示:
原來 git add 的最小粒度不是「文件」,而是 hunk(代碼塊)。git help add
而後查找 hunk
:
執行以下命令:
Git 會讓咱們有機會選擇對每個 hunk 作什麼樣的操做。這裏修改同一個位置,在一個 hunk 裏,根據提示咱們還要輸入 e
手工編輯 hunk。
將第 5 行刪除,保存退出,再看當前狀態:git diff --cached
:
git diff
:
目的達到了,過關:
正在特性分支上開發一個功能,被頭兒叫去修了一個緊急的 bug,修完後發現:媽蛋,那個特性分支叫啥?忘記了!
固然,做爲一個自律的程序員,通常是否是出現這樣的場景的。
這種狀況說明分支命名太沒有規律,或者分支太多,否則能夠經過 git branch
看一下,也能很快找到特性分支。
先看一下提示吧:
哦,原來有個命令叫 git reflog
,來看看幫助文檔 - git help reflog
,看起來這個命令很是強大,不過咱們這裏只用到簡單用法就能夠了:
上圖中第二行就顯示了咱們以前工做的特性分支。
有時代碼 push 到遠程倉庫後發現某一塊代碼有問題,咱們能夠經過 revert
命令將特定 commit 徹底恢復。
首先咱們要找到須要 revert 的 commit 的 hash:
徹底過程以下:
剛剛把最新的一次提交給毫無保留的扔掉,立刻就改了主意,怎麼辦?世界上有後悔藥嗎?
有的,只要進行 git 版本控制的一切都能找得回來。看下提示先:
提示說被咱們拋棄的那個 commit 像孤魂野鬼同樣在外遊蕩,尚未被鬼差送入地獄,咱們還能經過 git reflog
找到它的代號。
找到它的 Hash 後就經過 cherry-pick 將它找回來:
衝突合併是使用版本控制很是常見的了,竟然在這麼靠後的位置纔出來。
編輯衝突的文件 poem.txt,刪除 Git 添加的標識衝突的行。
別忘了,還要 git add poem.txt
而後 git commit
。
submodule 是 Git 組織大型項目的一種方式,一般可把第三方依賴做爲 submodule 包含進來,這個 submodule 自己也是一個獨立的 Git 項目。
最後這一關並不是測試使用 GitHub 的能力,而是指望你們貢獻代碼,包括增長更多關卡,修復 Bug 或者完善文檔。
我當初是準備翻譯中文版,結果工做量不小,拖的過久最後不了了之了。
但願你能對開源社區貢獻本身的一分力量!哪怕 Star
一下也是對做者莫大的鼓舞!
終於完結了,謝謝你這麼耐心一路跟着操練下來,但願你也像我同樣從這個遊戲裏收穫不少。
若是你喜歡這個系列文章,將它們分享給你的同事好友,也是對我最大的鼓勵!
若是有什麼意見反饋,歡迎評論留言,我必定回覆。
若是你發現有好的學習某技術的遊戲或教程,歡迎來到「軟件匠藝社區」 與你們分享!