我的具体案例是包含大量文本和IPv4地址的文本文档。我想删除除IP地址以外的所有内容。
我可以使用:vglobal
搜索([0-9]{1,3}\.){3}[0-9]{1,3}
并删除所有没有IP地址的行,但之后我只知道如何搜索整行并选择匹配的文本。有没有更简单的方法。
简而言之,我正在寻找一种不使用外部程序(如grep)来执行以下操作的方法:
grep --extended-regexp --only-matching --regexp="([0-9]{1,3}\.){3}[0-9]{1,3}"
从vim调用grep可能需要调整我的正则表达式(例如:remove \ v)。使用vim的增量搜索向我显示我的模式是正确的,我也不想在grep中验证我的正则表达式。
编辑:感谢彼得,这是我现在使用的功能。 (C是我通常在我的职能中使用的寄存器。)
"" Remove all text except what matches the current search result
"" The opposite of :%s///g (which clears all instances of the current search).
function! ClearAllButMatches()
let old = @c
let @c=""
%s//\=setreg('C', submatch(0), 'l')/g
%d _
put c
0d _
let @c = old
endfunction
Edit2:我把它作为一个接受范围的命令(但默认为整个文件)。
"" Remove all text except what matches the current search result. Will put each
"" match on its own line. This is the opposite of :%s///g (which clears all
"" instances of the current search).
function! s:ClearAllButMatches() range
let is_whole_file = a:firstline == 1 && a:lastline == line('$')
let old_c = @c
let @c=""
exec a:firstline .','. a:lastline .'sub//\=setreg("C", submatch(0), "l")/g'
exec a:firstline .','. a:lastline .'delete _'
put! c
"" I actually want the above to replace the whole selection with c, but I'll
"" settle for removing the blank line that's left when deleting the file
"" contents.
if is_whole_file
$delete _
endif
let @c = old_c
endfunction
command! -range=% ClearAllButMatches <line1>,<line2>call s:ClearAllButMatches()
答案 0 :(得分:11)
这种效果可以通过使用子替换特殊替换和setreg()
linewise
:let @a=""
:%s//\=setreg('A', submatch(0), 'l')/g
:%d _
:pu a
:0d _
或全部在一行中:
:let @a=""|%s//\=setreg('A', submatch(0), 'l')/g|%d _|pu a|0d _
概述:使用替换将每个匹配按顺序附加到寄存器“a”中,然后用寄存器“a”的内容替换整个缓冲区
说明:
let @a=""
清空我们将要附加到%s//\=setreg('A', submatch(0), 'l')/g
使用最后一个模式替换全局\=expr
将使用表达式submatch(0)
获取与之匹配的完整字符串setreg('A', submatch(0), 'l')
追加(注意:大写字母“a”)到@a匹配的字符串,但是按行%d _
将每一行删除到黑洞寄存器(又名@_)pu a
将@a的内容放入缓冲区0d _
删除第一行关注:
%s/<pattern>/\=setreg('A', submatch(0), 'l')/g
获取更多帮助
:h :s\=
:h :let-@
:h submatch()
:h setreg()
:h :d
:h :p
答案 1 :(得分:6)
假设<ip>
是你的正则表达式以匹配IP地址,我认为你可以做类似的事情:
:%s/.\{-}\(<ip>\).*/\1/g
其中\1
是第一个匹配的组(仅地址),.\{-}
用于非贪婪匹配。
答案 2 :(得分:5)
:set nowrapscan
:let @a=""
gg0qac/\v(\d{1,3}\.){3}\d{1,3}<CR><CR><Esc>//e+1<CR>@aq@adG
说明:
set nowrapscan
禁用了搜索«超过文件结尾»的能力。let @a=""
:清空一个寄存器。gg0
:转到第一行(gg)的第一列(0)。qa
:开始编写宏。c/{pattern}<CR>
:改变直到模式。c{motion}<CR><ESC>
:用换行符替换文字(此处{motion}
为/{pat}<CR>
)。//e+1<CR>
:搜索最后一个模式,将一个字符留在其末尾(包裹换行符,但如果您的行看起来像这样:IP<newline>IP
,则可能存在问题)。@a
:执行@a
个宏(录制时它是空的,但是当你完成它时会重复步骤1-7,直到它出现错误)。q
:结束录制@a
。@a
:执行@a
宏。dG
:删除到文件末尾。答案 3 :(得分:4)
简而言之,我正在寻找一种方法来做到这一点,而不必离开vim
足够简单:
:1,$! grep --extended-regexp --only-matching --regexp="([0-9]{1,3}\.){3}[0-9]{1,3}"
(虽然我实际上已经投了冰雪的替代答案)