今天碰到一个git的问题,在windows下用msysgit,总是提醒有文件修改,需要add to index。但是用git diff查看,没有特别的不同,只是显示整个文件都被修改。于是用git checkout -- <file>试图放弃本地修改,但是不起作用,还是提示文件修改过。
以前也看过类似的文章,怀疑是windows下换行符引起的问题。用ultraedit查看文件的二进制编码,发现本地文件的换行符为0D0A,这个格式是windows的回车+换行。登录到linux机器查看相同的文件,发现linux机器上的相同文件换行符为0A。基本上可以认为是换行符引起的问题。
原来,git有一个很重要的属性设置
git config --get core.autocrlf
这个属性如果设置为true,在执行git checkout/merge操作时会自动的将换行符从0A(lf)变成0D0A(crlf),而进行add/commit时则相反;
我的msysgit里面core.autocrlf的设置为true。每次git checkout都会先换成index里面的版本(0A换行),然后由于设置了core.autocrlf=true又会快速的切换成0D0A。于是就有了总是提示文件修改的故事。
解决办法
git config --set core.autocrlf false
实际上,git本身首先会看git repo里面的.gitattributes设置,例如下面的设置会自动的对java文件进行换行符转换。(crlf->lf)
*.java text=auto
其次才会看core.autocrlf设置。所以最好在git code repo里面设置一下
* text=auto
*.c text
*.h text
*.sln text eol=crlf
*.png binary
*.jpg binary
小结一下:
text 告诉git该文件是text,怎么处理要看eol设置(for 单独文件类型),或者core.eol设置(for 所有文件),以及其他设置选项。
text=auto 告诉git自动的进程换行符转换,git会自动判断文件是否是text文件类型。
[text] eol=crlf 告诉git对该类型文件进行换行符转换,转成crlf。
[text] eol=lf 告诉git对该类型文件进程换行符转换,转成lf
binary 告诉git对该类型文件不要进行换行符转换.
如果什么也没设置,看客户端core.autocrlf设置。