Git 記錄

Posted by Adam on August 24, 2022
### [Git 23 道情境題](https://gitexercises.fracz.com/committer/2f5t) ### [將 Git的預設編輯器更改為 Vim](https://medium.com/@racktar7743/git-%E5%B0%87-git%E7%9A%84%E9%A0%90%E8%A8%AD%E7%B7%A8%E8%BC%AF%E5%99%A8%E6%9B%B4%E6%94%B9%E7%82%BA-vim-41b5bcf024e1) ```bash git config --global core.editor "vim" ``` ### 停用 SSL 驗證 (Https) ```bash git config http.sslVerify "false" ``` ### 遠端功能 git remote 顯示 remote 詳情 ```bash git remote -v ``` 更新遠端分支的內容 (包含刪除) ```bash git remote update origin --prune ``` 強制上傳 ```bash git push -u --force remote_name branch_name ``` 移除遠端分支 ```bash git push --delete remote_name branch_name ``` 新增 origin 的遠端 ```bash git remote add origin git@github.com:User/UserRepo.git ``` 設定 origin 的 url ```bash git remote set-url --push --add origin git@github.com:poisondog/poisondog.commons.git ``` ### 分支功能 git branch ```bash # 顯示本地分支 git branch # 顯示遠端分支 git branch -r # 更新遠端分支的內容 (包含刪除) git remote update origin --prune # 移除分支 git branch -d branch_name # 分支呈現方式 # Show a pretty branch git log --all --graph --simplify-by-decoration --pretty=format:'%d' git log --all --graph --simplify-by-decoration --decorate --oneline alias glb='git log --all --graph --simplify-by-decoration --pretty=format:'%d'' alias glb='git log --all --graph --simplify-by-decoration --decorate --oneline' alias gld='git log --all --graph --decorate --oneline' # 分支重新命名 git branch -m oldname newname ``` ### 比對差異 ```bash # 列出當前分支和 <other-branch> 存在差異的檔案 git diff --name-only <other-branch> # 顯示簡要的統計信息,包括修改的文件數目和插入/刪除的行數。 git diff --stat ``` ### Commit 功能 git commit ```bash # 拆掉最後一個 commit git reset HEAD~1 # 拆掉最後兩個 commit git reset HEAD~2 # 更改最後 commit 的時間 git commit --amend --date="Wed Feb 16 14:00 2011 +0100" --no-edit # 修改最近一次 commit 的訊息 git commit --amend -m "message" # 修改以前的 commit 內容 git rebase -i HEAD^^ # 列出前兩個 commit 的修改內容 # mark the first commit with "edit" command # fix the typo in the file git add file.txt git rebase --continue # fix the typo in the commit message # 合併兩個 commit 成一個 commit git rebase -i HEAD^^ # "squash" or "fixup" the second commit #變更 commit 的順序 git rebase -i 開始的commit #編輯 commit 的順序 # 可選擇檔案中部分提交 git add -p file.txt # choose lines to include with 'y' git commit -m "First part of changes" git commit -am "The rest of the changed" ``` ```bash # 刪除 git 上的檔案但不刪除本地檔案 git rm --cached filename # for file git rm -r --cached foldername # for folder # 搜尋 log git log --grep=word ``` ### 暫存方法 ```bash git stash # 將目前的工作目錄暫存起來 git stash pop # 取回最後一次暫存的工作目錄 git stash apply # 取回最後一次暫存的工作目錄,但不刪除暫存 git stash list # 列出所有暫存的工作目錄 git stash clear # 清除所有暫存的工作目錄 # 暫存指定的檔案 git stash --patch git stash -p # 暫存區命名 git stash save feature-branch # 將暫存區命名為"feature-branch" ``` ### 強制切換,無視 merge 的衝突 ```bash # 強制切換,無視 merge 的衝突 git reset --hard origin/mybranch # 檔案名稱變更,避免 windows 檔名大小寫問題 git mv -f yOuRfIlEnAmE yourfilename # 以 anotherBranch 視為目前分支的基底開始 git rebase anotherBranch ``` rebase 轉植 ```bash git rebase --onto new-base-commit current-base-commit ``` 從歷程修改記錄中強制重新設定 branch ```bash git reflog #所有歷程修改記錄 git reset --hard HEAD@{1} ``` 僅針對 path_to_file 更換成 source_branch 的內容 ```bash git checkout -p source_branch -- path_to_file ``` 搜尋修改內容符合 word 的commit ```bash git log -Sword ``` ```bash # git bisect 為 git 提供的 debug 方法 # git debug 找出第一個出錯的方法 git bisect start # 從目前的 commit 開始搜尋 git bisect bad HEAD # HEAD 標記為錯誤的情況 git bisect good 1.0 # 1.0 的 Tag 標記是正確的情況 git bisect run "./test.sh" #利用 test.sh 當作檢測腳本,回傳0為正確情況,其它則作為錯誤情況。 git bisect run sh -c "openssl enc -base64 -A -d < home-screen-text.txt | grep -v jackass" ``` ### [終於搞懂如何 revert merge commit](https://medium.com/@kurosean/%E7%B5%82%E6%96%BC%E6%90%9E%E6%87%82%E5%A6%82%E4%BD%95-revert-merge-commit-a034b9e69fef) ``` * 911ee44 update setting * d3beb55 Merge branch 'master' of ssh://someplace |\ | * 133f8d1 add someip to access 8080 * | 24183d5 update tensorflow-gpu yml * | 87f26f7 update |/ * f6eebd0 add knockd.conf * 276f6d3 update push script git revert d3beb55 -m 1 # revert 133f8d1 git revert d3beb55 -m 2 # revert 24183d5 and 87f26f7 ``` ### [【狀況題】檢視特定檔案的 Commit 紀錄](https://gitbook.tw/chapters/using-git/view-log-of-a-file) ``` git log welcome.html git log -p welcome.html ``` ### [解決合併時衝突的工具](https://gist.github.com/karenyyng/f19ff75c60f18b4b8149) ```bash git mergetool ``` ### git cherry-pick `git cherry-pick` 命令默認用於單一 commit 的複製,但實際上你可以指定多個 commit 進行複製。以下是一些基本的使用方式: 1. **單一 commit 的 cherry-pick:** ```bash git cherry-pick <commit-hash> ``` 2. **多個 commit 的 cherry-pick:** ```bash git cherry-pick <commit-hash-1> <commit-hash-2> <commit-hash-3> ``` 這樣可以按照指定的順序將多個 commit 複製到目標分支。如果其中某個 commit 有衝突,`git cherry-pick` 將暫停並等待你解決衝突。 3. **範圍內的 commit 的 cherry-pick:** 你也可以指定 commit 範圍,例如,複製 commit A 到 commit B 之間的所有 commit: ```bash git cherry-pick A^..B ``` 這將包括 commit A 和 commit B。 總的來說,`git cherry-pick` 是可以處理單一 commit,多個 commit 或者 commit 範圍的,只需提供相應的 commit hash 或 commit 範圍即可。當進行多個 commit 的 cherry-pick 時,可能會遇到合併衝突,需要解決這些衝突後再繼續進行。 ```bash git cherry-pick feature-a git cherry-pick feature-b git cherry-pick feature-c # resolve merge conflict git add -A git cherry-pick --continue ``` ### [git rebase](https://is.gd/m63eNw) 假設有三分支 ``` * a1c877f (beta) F * 11b5bb1 E * cbaa785 D | * 557dbf2 (alpha) C | * 1886c7a B | * c2e94d4 A |/ * 985e5d8 (HEAD -> master) init ``` ```bash git rebase --onto 557dbf2 985e5d8 beta # 557dbf2 基礎點 (枝幹) # 985e5d8 嫁接起點 (接穗起點,不包含此 commit) # beta 嫁接終點 (接穗終點,包含此 commit) * 315ac69 (HEAD -> beta) F * 05705b2 E * 5cc80e0 D * 557dbf2 (alpha) C * 1886c7a B * c2e94d4 A * 985e5d8 (master) init ``` ``` git rebase --onto c2e94d4 985e5d8 a1c877f # c2e94d4 基礎點 (枝幹) # 985e5d8 嫁接起點 (接穗起點,不包含此 commit) # a1c877f 嫁接終點 (接穗終點,包含此 commit) * e489c4e (HEAD) F * 723e137 E * 69b6ee6 D | * a1c877f (beta) F | * 11b5bb1 E | * cbaa785 D | | * 557dbf2 (alpha) C | | * 1886c7a B | |/ |/| * | c2e94d4 A |/ * 985e5d8 (master) init ``` ### [git rebase --onto](https://womanonrails.com/git-rebase-onto) ``` git rebase --onto F D ``` ``` Before After A---B---C---F---G (branch) A---B---C---F---G (branch) \ \ D---E---H---I (HEAD my-branch) E'---H'---I' (HEAD my-branch) ``` ``` git rebase --onto F D I ``` ``` Before After A---B---C---F---G (branch) A---B---C---F---G (branch) \ | \ D---E---H---I (HEAD my-branch) | E'---H'---I' (HEAD) \ D---E---H---I (my-branch) ``` ``` git rebase --onto F D H ``` ``` Before After A---B---C---F---G (branch) A---B---C---F---G (branch) \ | \ D---E---H---I (HEAD my-branch) | E'---H' (HEAD) \ D---E---H---I (my-branch) ``` ### git blame 在使用git指令查看歷史紀錄時,可以使用`git blame`指令來顯示每一行是由哪個使用者異動的。透過`git blame`可以看到每一行的最後修改者、修改的commit hash、修改的時間。 ```bash git blame <檔案名稱> ``` 另外,也可以使用圖形化界面的git工具來查看歷史紀錄和每一行的最後修改者,例如使用`git log --graph --all`指令來顯示commit的圖形化歷史紀錄。 要在 Vim 中查看 git blame 的內容,可以使用以下步驟進行設定: 1. 確保你已經安裝了 vim-fugitive 插件,這個插件可以讓你在 Vim 中使用 Git 的一些功能。 2. 打開一個代碼文件,在 Normal 模式下輸入 :Gblame,然後按 Enter。這將打開一個新的分割窗口,其中會顯示 git blame 的內容。 3. 在 git blame 的內容中,可以使用 j 和 k 鍵來上下滾動。如果需要退出 git blame 模式,可以按 q 鍵。 透過這些步驟,你就可以在 Vim 中輕鬆地查看 git blame 的內容了。 ### 修改 commit 訊息 當您需要大量修改 commit 訊息時,可以使用以下步驟進行 git rebase: 1. 開始互動式 rebase 模式: ```bash git rebase -i HEAD~n ``` 這裡的 n 代表您希望修改的 commit 數量。執行後,您會看到一個類似如下的互動式介面: ``` pick 0123456 Commit message 1 pick 7890123 Commit message 2 ... ``` 2. 更改您想要修改的 commit 訊息: 在互動式介面中,將 `pick` 改為 `reword` 或 `r` 以指示您希望修改該 commit 訊息。儲存並關閉編輯器。 3. 逐一修改 commit 訊息: 系統將會逐一顯示您指定的 commit,讓您可以修改其 commit 訊息。在每個 commit 中,您可以修改訊息並儲存,然後系統會自動繼續 rebase 操作。 4. 完成 rebase: 當您完成所有 commit 訊息的修改後,請按照系統指示完成 rebase 操作。您可能需要處理任何可能發生的衝突,然後使用 `git rebase --continue` 來繼續進行 rebase。 透過這些步驟,您可以大量修改多個 commit 的訊息,讓您的歷史紀錄更加清晰與易讀。希望這個解釋對您有幫助!如果有任何疑問,請隨時告訴我。