2019年11月1日 星期五

[Git 文章收集] 【狀況題】如何在 Git 裡刪除檔案或變更檔名?

Source From Here
在 Git 裡,不管是刪除檔案或是變更檔名,對 Git 來說都是一種「修改」。

刪除檔案

直接砍
你可以使用系統指令 rm 或是檔案總管之類的工具來刪除檔案,例如:
# rm welcome.html
rm: remove regular file ‘welcome.html’? y

# git status
# On branch dev
Changes not staged for commit:
# (use "git add/rm ..." to update what will be committed)
# (use "git checkout -- ..." to discard changes in working directory)
#
deleted: welcome.html
#
no changes added to commit (use "git add" and/or "git commit -a")

可以看到 welcome.html 這個檔案目前的狀態是 deleted。如果你確定這是你想做的,就可以把這次的「修改」加到暫存區:
# git rm welcome.html
rm 'welcome.html'

# git status
# On branch dev
Changes to be committed:
# (use "git reset HEAD ..." to unstage)
#
deleted: welcome.html
#

它現在的狀態是 staged,而且已被加至暫存區,所以接下來就可以進行 Commit 了。我知道「把刪除檔案加到暫存區」這件事感覺有點不太直覺,就把「刪除檔案」也當做是一種「修改」看待就行了。

請 Git 幫你砍
像這樣先 rm 刪除然後再 git rm 加入暫存區的兩段式動作,事實上可以直接使用 git rm 指令. 首先來救回 welcome.html 檔案:
# git reset HEAD
# git status
# On branch dev
Changes not staged for commit:
# (use "git add/rm ..." to update what will be committed)
# (use "git checkout -- ..." to discard changes in working directory)
#
deleted: welcome.html
#
no changes added to commit (use "git add" and/or "git commit -a")


# git checkout -- welcome.html
# cat welcome.html
Hello

接著直接使用 git rm 移除:
# git rm welcome.html
rm 'welcome.html'

# git status
# On branch dev
Changes to be committed:
# (use "git reset HEAD ..." to unstage)
#
deleted: welcome.html
#

它就直接在暫存區了,可以少做一個步驟。

加上 –cached 參數
不管是系統的 rm 或是 git rm 指令,都會真的把這個檔案從工作目錄裡刪掉,但如果只是「我不是真的想把這個檔案刪掉,只是不想讓這個檔案再被 Git 控管了」的話,可以加上 --cached 參數:
// 請先參考前面步驟將 welcome.html 救回
# git log --oneline | head -n 1
d1dcb5c Add welcome.html

// 檢視 welcome.html 存在且 committed
# cat welcome.html
Hello

// 從 staging 中刪除 welcome.html
# git rm --cached welcome.html
rm 'welcome.html'

// 檔案系統的 welcome.html 依舊存在
# cat welcome.html
Hello

// 提交 welcome.html 的 commit 還在
# git log --oneline | head -n 1
d1dcb5c Add welcome.html

# git status
# On branch dev
Changes to be committed:
# (use "git reset HEAD ..." to unstage)
#
deleted: welcome.html
#
Untracked files:
# (use "git add ..." to include in what will be committed)
#
welcome.html

welcome.html 的狀態從原本已經在 Git 目錄裡的 tracked 變成 Untracked 了。你依舊需要 commit 來將 welcome.html 從 Git 版本管控移除:
# git commit -m 'Remove welcome.html'
[dev 9ea74ee] Remove welcome.html
1 file changed, 1 deletion(-)
delete mode 100644 welcome.html


# git status
# On branch dev
Untracked files:
# (use "git add ..." to include in what will be committed)
#
welcome.html
nothing added to commit but untracked files present (use "git add" to track)

接著你可以參考 【狀況題】有些檔案我不想放在 Git 裡面 讓 welcome.html 不再出現 untracked file 中.


變更檔名

直接改名
跟刪除檔案一樣,變更檔名也是一種「修改」,所以操作上其實也是差不多的:
# mv hello.html world.html

# git status
# On branch dev
Changes not staged for commit:
# (use "git add/rm ..." to update what will be committed)
# (use "git checkout -- ..." to discard changes in working directory)
#
deleted: hello.html
#
Untracked files:
# (use "git add ..." to include in what will be committed)
#
world.html
no changes added to commit (use "git add" and/or "git commit -a")

雖然只是改檔名,但對 Git 來說會被認為是兩個動作,一個是刪除 hello.html 檔案,一個是新增 world.html 檔案(變成 Untracked 狀態)。接著繼續使用 git add 指令把這些異動加至暫存區:
# git add --all

# git status
# On branch dev
Changes to be committed:
# (use "git reset HEAD ..." to unstage)
#
renamed: hello.html -> world.html
#

因為檔案的內容沒有改變,Git 猜得出來這個只是單純的改名字,所以現在它的狀態變成 renamed 了。

請 Git 幫你改名
跟前面的 git rm 一樣,Git 也有提供類似的指令 git mv 可以讓你少做一步:
# git mv hello.html world.html

# git status
# On branch dev
Changes to be committed:
# (use "git reset HEAD ..." to unstage)
#
renamed: hello.html -> world.html
#

【冷知識】其實 Git 根本不在乎你的檔案叫什麼名字!
Git 是根據檔案的「內容」去算出那個 SHA-1 的值,所以 Git 不是很在乎你的檔案叫什麼名字,只在乎檔案的內容是什麼。所以當你進行更改檔名的時候,Git 並沒有為此做出一個新的 Blob 物件,而僅是指向原來舊的那顆 Blob 物件。但因為檔名變了,所以會為此做出一顆新的 Tree 物件喔。如果這些 Git 物件還不清楚它們是幹嘛的,在後面的「【超冷知識】在 .git 目錄裡有什麼東西?Part 1」章節會有詳細的說明

沒有留言:

張貼留言

[Git 常見問題] error: The following untracked working tree files would be overwritten by merge

  Source From  Here 方案1: // x -----删除忽略文件已经对 git 来说不识别的文件 // d -----删除未被添加到 git 的路径中的文件 // f -----强制运行 #   git clean -d -fx 方案2: 今天在服务器上  gi...