substitution, transform string to list?

13 views
Skip to first unread message

meine

unread,
Jul 17, 2020, 5:10:25 AM7/17/20
to vim...@googlegroups.com
Hi,

I encountered a nice question to transform some data, and are puzzled
how to do this in Vim. Preferably with the least possible keystrokes,
substitution above a macro. The complete set to transform is some 80k
lines and the first word might differ per line.

Who can help here?

The data to transform:

ABBEEL;1872/253;1882/576;1886/1925;1887/603;1887/1813;1894/1444;1898/3449
ABBEELS;1888/1401;1889/562;1891/2371;1899/3088;1902/732;1903/1509;1904/1317

should become:

ABBEEL;1872/253
ABBEEL;1882/576
ABBEEL;1886/1925
ABBEEL;1887/603
ABBEEL;1887/1813
ABBEEL;1894/1444
ABBEEL;1898/3449
ABBEELS;1888/1401
ABBEELS;1889/562
ABBEELS;1891/2371
ABBEELS;1899/3088
ABBEELS;1902/732
ABBEELS;1903/1509
ABBEELS;1904/1317

Regular substitution `:s/;/\rABBEEL;/g' only works for each line and the
first word has to be typed by hand. I tried using wildcards like `\w\+'
and `.*' but only got a messy result (source is here:
https://stackoverflow.com/questions/10336609/is-it-possible-to-use-find-and-replace-on-a-wildcard-string-in-vim)

TIA,

//meine

Chris Willis

unread,
Jul 17, 2020, 8:01:30 AM7/17/20
to vim...@googlegroups.com
How about a perl one-liner:

perl -F; -nae "$a = shift(@F) ; for (@F){ chomp; print(qq($a;$_\n));}"
vtd.txt

where vtd.txt is your data file and you can redirect the output somewhere
other than the current STDOUT.

Chris Willis

Tim Chase

unread,
Jul 17, 2020, 8:46:17 AM7/17/20
to meine, vim...@googlegroups.com
On 2020-07-17 11:10, meine wrote:
> The data to transform:
>
> ABBEEL;1872/253;1882/576;1886/1925;1887/603;1887/1813;1894/1444;1898/3449
> ABBEELS;1888/1401;1889/562;1891/2371;1899/3088;1902/732;1903/1509;1904/1317
>
> should become:
>
> ABBEEL;1872/253
> ABBEEL;1882/576
> ABBEEL;1886/1925
> ABBEEL;1887/603
> ABBEEL;1887/1813
> ABBEEL;1894/1444
> ABBEEL;1898/3449
> ABBEELS;1888/1401
> ABBEELS;1889/562
> ABBEELS;1891/2371
> ABBEELS;1899/3088
> ABBEELS;1902/732
> ABBEELS;1903/1509
> ABBEELS;1904/1317

You can do it in two passes, one to prefix each item on its own line:

:%s/\%(^\([^;]*\).*\)\@<=\(;[^;]*\)/\r\1\2/g

and then a second command to delete all the remnant lines (ones that
are just the prefix which don't have a ";" in them):

:v/;/d

-tim



meine

unread,
Jul 17, 2020, 10:05:34 AM7/17/20
to Tim Chase, vim...@googlegroups.com
Thanks a lot!

//meine
Reply all
Reply to author
Forward
0 new messages