我在Stack Overflow上看了几个关于如何将空格转换为制表符而没有找到我需要的问题。关于如何将制表符转换为空格,似乎有更多的问题,但我正试图做相反的事情。

在Vim中,我尝试过:retab和:retab!没有运气,但我相信这些实际上是用来从制表符到空格的。

我在命令提示符下尝试了展开和取消展开,但没有任何运气。

以下是该文件:

http://gdata-python-client.googlecode.com/hg-history/a9ed9edefd61a0ba0e18c43e448472051821003a/samples/docs/docs_v3_example.py

如何使用Vim或shell将前导空格转换为制表符?


使用Vim扩展所有前导空格(比'tabstop'更宽),使用retab是正确的,但首先确保'expandtab'被重置(:verbose set ts?外星人吗?是你的朋友)。retab需要一个范围,所以我通常指定%表示“整个文件”。

:set tabstop=2      " To match the sample file
:set noexpandtab    " Use tabs, not spaces
:%retab!            " Retabulate the whole file

在做这样的事情之前(特别是Python文件!),我通常设置“list”,这样我就可以看到空白和更改。

我在我的.vimrc中有以下映射:

nnoremap    <F2> :<C-U>setlocal lcs=tab:>-,trail:-,eol:$ list! list? <CR>

1 -如果你有空格,想要制表符。

首先,您需要决定单个制表符将包含多少个空格。也就是说,假设你的行前导有4个空格,或者8…然后你会意识到制表符应该是4个空格。现在有了这些信息,你可以:

:set ts=4
:set noet
:%retab!

这里有个问题!这个命令序列将查找所有文本,而不仅仅是行首的空格。这意味着像“嘿,␣这个␣␣␣␣是␣4个␣空格”这样的字符串将变成“嘿,␣这个⇥是␣4个␣空格”,但它不是!这是账单!

为了解决这个小问题,我建议搜索,而不是重新标签。

:%s/^\(^I*\)␣␣␣␣/\1^I/g

这个搜索将在整个文件中查找以任意数量的制表符开始,后面跟着4个空格的任何行,并将其替换为它找到的任意数量的制表符加1。

不幸的是,这将不会立即运行!

首先,文件的行以空格开头。然后搜索将只转换前4个空格为制表符,并让下面的…

需要重复执行该命令。多少次?直到你找到一个没有发现的模式。我还想不出一个方法来自动化这个过程。但如果你这样做了:

`10@:`

你可能已经完成了。该命令将重复上一次搜索/替换操作10次。你的程序不太可能有这么多缩进。如果有,再重复一次@@。

现在,我们来完成这个答案。我知道你想要的恰恰相反,但你永远不知道什么时候需要挽回。

2 -你有制表符,想要空格。

首先,决定要将制表符转换为多少个空格。假设你想让每个制表符是2个空格。然后你要做:

:set ts=2
:set et
:%retab!

字符串也会有同样的问题。但是在字符串中不使用硬制表符是更好的编程风格,你实际上在这里做了一件好事。如果确实需要字符串中的制表符,请使用\t。


:%s/\(^\s*\)\@<=    /\t/g

转换:搜索4个连续空格(在=字符之后)的每个实例,但前提是直到该点的整行都是空格(这使用了零宽度向后查找断言\@<=)。用制表符替换每个找到的实例。


简单Python脚本:

import os

SOURCE_ROOT = "ROOT DIRECTORY - WILL CONVERT ALL UNDERNEATH"

for root, dirs, files in os.walk(SOURCE_ROOT):
    for f in files:
        fpath = os.path.join(root,f)
        assert os.path.exists(fpath)
        data = open(fpath, "r").read()
        data = data.replace("    ", "\t")
        outfile = open(fpath, "w")
        outfile.write(data)
        outfile.close()

Linux:使用unexpand(和expand)

这里有一个非常好的解决方案:https://stackoverflow.com/a/11094620/1115187,主要是因为它使用*nix-utilities:

Unexpand - Spaces ->选项卡 Expand - tabs ->空格

Linux:自定义脚本

我最初的回答

在。/app文件夹中的所有.py文件中替换4-空格缩进(脚本中有两个{4})的Bash代码片段(递归):

find ./app -iname '*.py' -type f \
    -exec awk -i inplace \
    '{ match($0, /^(( {4})*)(.*?)$/, arr); gsub(/ {4}/, "\t", arr[1]) }; { print arr[1] arr[3] }' {} \; 

它不会在中间或结尾修改4-空格。

在Ubuntu 16.0x和Linux Mint 18下测试


将所有空格改为制表符 : % s / \ / \ t / g


在我的例子中,我有多个空格(字段由一个或多个空格分隔),我想用一个制表符替换它们。下面的人做到了:

:% s/\s\+/\t/g

如果您安装了GNU coreutils,请考虑%!不展开—仅优先或对于4个空格制表符,考虑%!Unexpand -t 4——first-only(——first-only是为了防止你意外地用——all调用Unexpand)。

注意,这将只替换指定制表位之前的空格,而不是后面的空格;在vim中,你不会看到任何视觉上的差异,除非你更字面地显示标签;例如,我的~/。vimrc包含集合列表listchars=tab:┈(我怀疑这就是为什么你认为unexpand不起作用)。


使用Vim重新标签一组文件(例如,所有的*。从2个空格到4个空格,你可以尝试从命令行:

find . -name '*.ts' -print0 | xargs -0 -n1 vim -e '+set ts=2 noet | retab! | set ts=4 et | retab | wq'

它所做的是使用find将所有匹配的文件传递给xargs (find上的-print0选项与xargs上的-0选项一起工作,以便处理名称中带有空格的文件)。

Xargs以ex模式(-e)对每个文件运行vim,执行给定的ex命令(实际上是几个命令),将现有的前导空格更改为制表符,重置制表位并将制表符更改为空格,最后保存并退出。

在ex模式下运行可以防止:Vim:警告:每个文件的输入不是来自一个终端。


在GNU coreutils的shell中: unexpand -t2——first-only originalFile > newFile 解释: Unexpand:将空格转换为制表符的命令 -t2:把2个空格变成1个制表符 ——first-only:只修改行首的空白(包括混合制表符和空格),不修改行中和行尾的空白

它的倒数是expand它只需要一个简单的-i而不是-first-only, -t2不变

Johnsyweb流行的绿色检查答案是完全错误的,因为它修改了到处的空白!!

附:这篇文章的疯狂格式并不能反映内容的质量。它反映了如何stackoverflow不支持我的浏览器。谢谢你!