### [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 的訊息,讓您的歷史紀錄更加清晰與易讀。希望這個解釋對您有幫助!如果有任何疑問,請隨時告訴我。