我读到在Git中重命名文件时,你应该提交任何更改,执行重命名,然后展示重命名的文件。Git将从内容中识别文件,而不是将其视为一个新的未跟踪的文件,并保留更改历史。

然而,今晚这样做,我最终恢复到git mv。

> $ git status
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#    modified:   index.html
#

我重命名我的样式表在Finder从iphone.css到mobile.css:

> $ git status
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#    modified:   index.html
#
# Changed but not updated:
#   (use "git add/rm <file>..." to update what will be committed)
#   (use "git checkout -- <file>..." to discard changes in working directory)
#
#    deleted:    css/iphone.css
#
# Untracked files:
#   (use "git add <file>..." to include in what will be committed)
#
#    css/mobile.css

所以Git现在认为我删除了一个CSS文件,并添加了一个新的文件。这不是我想要的。让我们撤销重命名,让Git来完成这项工作。

> $ git reset HEAD .
Unstaged changes after reset:
M    css/iphone.css
M    index.html

我又回到了开始的地方:

> $ git status
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#    modified:   index.html
#

让我们用git mv代替:

> $ git mv css/iphone.css css/mobile.css
> $ git status
# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#    renamed:    css/iphone.css -> css/mobile.css
#
# Changed but not updated:
#   (use "git add <file>..." to update what will be committed)
#   (use "git checkout -- <file>..." to discard changes in working directory)
#
#    modified:   index.html
#

看起来我们没事了。那么,为什么Git在我第一次使用Finder时没有识别出这个重命名呢?


当前回答

您必须将这两个修改后的文件添加到索引中,Git才能将其识别为移动。

mv old new和git mv old new之间的唯一区别是git mv也将文件添加到索引中。

mv old new然后git add -A也可以工作。

请注意,不能只使用git add .,因为它不会向索引添加删除。

参见“git add -A”和“git add”的区别。

其他回答

最好的办法是自己去试试。

mkdir test
cd test
git init
touch aaa.txt
git add .
git commit -a -m "New file"
mv aaa.txt bbb.txt
git add .
git status
git commit --dry-run -a

现在git状态和git提交——dry-run -a显示两个不同的结果,其中git状态显示bbb.txt作为一个新文件/ aaa.txt被删除,而——dry-run命令显示实际的重命名。

~/test$ git status

# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#   new file:   bbb.txt
#
# Changes not staged for commit:
#   (use "git add/rm <file>..." to update what will be committed)
#   (use "git checkout -- <file>..." to discard changes in working directory)
#
#   deleted:    aaa.txt
#


/test$ git commit --dry-run -a

# On branch master
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#   renamed:    aaa.txt -> bbb.txt
#

现在去办理登机手续。

git commit -a -m "Rename"

现在您可以看到文件实际上已重命名,并且git status中显示的内容是错误的,至少在本例中是这样。下划线GIT实现可能会将这两个命令分开处理。

这个故事的寓意:如果你不确定你的文件是否被重命名了,发出一个“git commit -dry-run -a”。如果它显示文件已重命名,那么就可以开始了。

对于git mv手册页说

索引在成功完成后更新, […]

因此,首先,您必须自己更新索引 (使用git添加mobile.css)。但是git状态 仍然会显示两个不同的文件:

$ git status
# On branch master
warning: LF will be replaced by CRLF in index.html
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#       modified:   index.html
#       new file:   mobile.css
#
# Changed but not updated:
#   (use "git add/rm <file>..." to update what will be committed)
#   (use "git checkout -- <file>..." to discard changes in working directory)
#
#       deleted:    iphone.css
#

你可以通过运行git commit——dry-run -a来得到不同的输出 期望:

Tanascius@H181 /d/temp/blo (master)
$ git commit --dry-run -a
# On branch master
warning: LF will be replaced by CRLF in index.html
# Changes to be committed:
#   (use "git reset HEAD <file>..." to unstage)
#
#       modified:   index.html
#       renamed:    iphone.css -> mobile.css
#

我不能确切地告诉你为什么我们会看到这些差异 在git状态和git提交之间——dry-run -a,但是 下面是莱纳斯的提示:

Git根本不关心整体 在内部“重命名检测”,以及您拥有的任何提交 所做的重命名完全独立于 然后我们使用启发式来显示重命名。

试运行使用真正的重命名机制,而 Git状态可能不会。

在windows 10的git 2.33上测试

重命名文件夹和文件你需要在资源管理器。 Git添加。 Git commit -m "提交信息" git推

Git将检测重命名。你可以通过运行git状态来检查(提交后)

让我们从Git的角度来考虑您的文件。

请记住,Git不会跟踪任何关于文件的元数据

您的存储库具有(以及其他)

$ cd repo
$ ls
...
iphone.css
...

它在Git的控制下:

$ git ls-files --error-unmatch iphone.css &>/dev/null && echo file is tracked
file is tracked

用以下方法进行测试:

$ touch newfile
$ git ls-files --error-unmatch newfile &>/dev/null && echo file is tracked
(no output, it is not tracked)
$ rm newfile

当你这样做时

$ mv iphone.css mobile.css

从Git的角度来看,

没有任何iphone.css(它被删除了-git对此发出警告-)。 有一个新文件mobile。css。 这些文件完全不相关。

因此,Git会对它已经知道的文件(iphone.css)和它检测到的新文件(mobile.css)提出建议,但只有当文件位于index或HEAD中时,Git才会开始检查它们的内容。

目前,“iphone.css deletion”和“mobile.css”都不在索引中。

将iphone.css delete添加到索引中:

$ git rm iphone.css

Git告诉你到底发生了什么(iphone.css被删除。没有别的事情发生):

然后添加新的文件mobile.css:

$ git add mobile.css

这次,删除文件和新建文件都在索引中。现在Git检测到上下文是相同的,并将其公开为重命名。事实上,如果文件有50%的相似度,它会将其检测为重命名,允许您稍微更改mobile.css,同时保持操作的重命名。

这在git diff上是可复制的,现在你的文件在索引中,你必须使用-cached。编辑mobile.css一点,将其添加到索引,并查看以下区别:

$ git diff --cached

and

$ git diff --cached -M

-M是git diff的“检测重命名”选项。-M代表-M50%(50%或更多的相似度将使git表示为重命名),但如果你经常编辑mobile.css文件,你可以将其减少为-M20%(20%)。

你没有展示你的搜索结果。我相信如果你通过Finder移动然后git添加css/mobile。css;git rm css/iphone.css, git会计算新文件的哈希值,然后才意识到文件的哈希值匹配(因此这是一个重命名)。