303

所以我在一家 PHP 商店工作,我们都使用不同的编辑器,而且我们都必须在 Windows 上工作。我使用 vim,店里的每个人都在抱怨每当我编辑文件时,底部都会有一个换行符。我四处搜索,发现这是 vi 和 vim 的记录行为......但我想知道是否有某种方法可以禁用此功能。(如果我可以为特定的文件扩展名禁用它,那将是最好的)。

如果有人知道这一点,那就太好了!

4

13 回答 13

440

对于vim7.4+,您可以使用(最好在您的 .vimrc 上)(感谢罗泽轩的最后一点消息!):

:set nofixendofline

现在关于旧版本的vim.

即使文件最后已经用新行保存:

vim -b file 并在 vim 中:

:set noeol
:wq

完毕。

或者,您可以在 vim 中打开文件:e ++bin file

还有另一种选择:

:set binary
:set noeol
:wq

为什么我需要二进制模式下的 vim 以使“noeol”工作?

于 2013-04-19T22:42:26.383 回答
23

将以下命令添加到您的 .vimrc 以关闭行尾选项:

autocmd FileType php setlocal noeol binary fileformat=dos

但是,PHP 本身会忽略最后一个行尾——这应该不是问题。我几乎可以肯定,在您的情况下,还有其他东西正在添加最后一个换行符,或者可能与 windows/unix 行尾类型(\n\r\n等)混淆。

更新:

另一种解决方案可能是将此行添加到您的 .vimrc 中:

set fileformats+=dos
于 2009-07-02T00:59:26.203 回答
18

如果您使用 Git 进行源代码控制,还有另一种方法可以解决这个问题。受此处答案的启发,我编写了自己的过滤器以在gitattributes文件中使用。

要安装此过滤器,请将其保存noeol_filter在您的某个位置$PATH,使其可执行,然后运行以下命令:

git config --global filter.noeol.clean noeol_filter
git config --global filter.noeol.smudge cat

要开始仅为您自己使用过滤器,请将以下行放入您的$GIT_DIR/info/attributes:

*.php filter=noeol

这将确保您不会在.php文件中的 eof 处提交任何换行符,无论 Vim 做什么。

现在,脚本本身:

#!/usr/bin/python

# a filter that strips newline from last line of its stdin
# if the last line is empty, leave it as-is, to make the operation idempotent
# inspired by: https://stackoverflow.com/questions/1654021/how-can-i-delete-a-newline-if-it-is-the-last-character-in-a-file/1663283#1663283

import sys

if __name__ == '__main__':
    try:
        pline = sys.stdin.next()
    except StopIteration:
        # no input, nothing to do
        sys.exit(0)

    # spit out all but the last line
    for line in sys.stdin:
        sys.stdout.write(pline)
        pline = line

    # strip newline from last line before spitting it out
    if len(pline) > 2 and pline.endswith("\r\n"):
        sys.stdout.write(pline[:-2])
    elif len(pline) > 1 and pline.endswith("\n"):
        sys.stdout.write(pline[:-1])
    else:
        sys.stdout.write(pline)
于 2011-06-28T08:04:31.393 回答
13

这个选项我没试过,但是vim帮助系统中给出了以下信息(即help eol):

'endofline' 'eol'   boolean (default on)
            local to buffer
            {not in Vi}

When writing a file and this option is off and the 'binary' option
is on, no <EOL> will be written for the last line in the file.  This
option is automatically set when starting to edit a new file, unless
the file does not have an <EOL> for the last line in the file, in
which case it is reset.  

通常您不必设置或重置此选项。当 'binary' 关闭时,写入文件时不使用该值。当 'binary' 开启时,它用于记住文件中最后一行的 a 的存在,以便在您写入文件时可以保留原始文件中的情况。但如果你愿意,你可以改变它。

您可能也对上一个问题的答案感兴趣:“为什么文件应该以换行符结尾”。

于 2009-06-26T19:16:12.333 回答
10

我在 Vim wiki 上添加了一个针对类似(尽管不同)问题的提示:

http://vim.wikia.com/wiki/Do_not_auto-add_a_newline_at_EOF

于 2010-12-31T02:48:44.983 回答
8

好的,你在 Windows 上会使事情复杂化;)

由于 'binary' 选项重置了 'fileformat' 选项(并且使用 'binary' 设置写入总是使用 unix 行结尾),让我们拿出大锤子,在外部进行!

为 BufWritePost 事件定义一个自动命令(:help autocommand)怎么样?每次写入整个缓冲区后都会执行此自动命令。在这个自动命令中,调用一个小的外部工具(php、perl 或任何脚本)来去除刚刚写入的文件的最后一个换行符。

所以这看起来像这样,并会进入你的 .vimrc 文件:

autocmd!   "Remove all autocmds (for current group), see below"
autocmd BufWritePost *.php !your-script <afile>

如果这是您第一次处理自动命令,请务必阅读有关自动命令的整个 vim 文档。有一些注意事项,例如建议删除 .vimrc 中的所有自动命令,以防 .vimrc 可能被多次获取。

于 2009-07-02T07:22:29.180 回答
6

我已经使用 Perl 和 Python 后处理实现了 Blixtor 的建议,或者在 Vim 中运行(如果它是使用这种语言支持编译的),或者通过外部 Perl 脚本。它在 vim.org 上作为PreserveNoEOL 插件提供。

于 2013-04-26T12:14:34.947 回答
5

vimv7.4 开始,您可以使用

:set nofixendofline

这里有一些关于这个变化的信息:http: //ftp.vim.org/vim/patches/7.4/7.4.785

于 2019-08-01T17:57:04.627 回答
3

也许你可以看看他们为什么抱怨。如果一个 php 文件在结尾 ?> 之后有一个换行符,php 会将它作为页面的一部分输出。除非您在包含文件后尝试发送标头,否则这不是问题。

但是,php 文件末尾的 ?> 是可选的。没有结尾?>,文件末尾的换行符没有问题。

于 2009-06-28T17:04:03.223 回答
3

尝试添加.vimrc

set binary
于 2013-05-30T05:43:58.733 回答
2

我想我找到了比公认答案更好的解决方案。替代解决方案对我不起作用,我不想一直在二进制模式下工作。幸运的是,这似乎完成了工作,我还没有遇到任何令人讨厌的副作用:在文本文件末尾保留缺少的行尾。我刚刚将整个内容添加到我的 ~/.vimrc 中。

于 2013-08-16T05:47:01.600 回答
1

您是否可以使用特殊命令来保存这些文件?

如果你这样做 :set binary, :w 和 :set nobinary 如果没有新行,则文件将在没有换行符的情况下写入。

当然,这个命令序列可以放入用户定义的命令或映射中。

于 2009-07-01T09:19:51.970 回答
0

我发现这个 vimscript 插件对这种情况很有帮助。

Plugin 'vim-scripts/PreserveNoEOL'

或者在github 上阅读更多内容

于 2014-07-12T00:57:55.180 回答