bash to merge lines in a file

跟風遠走 提交于 2020-01-04 16:58:38

问题


I want convert this text:

qa-ops01.mysite.com
/dev/mapper/sys-home   58G   26G   30G  47% /home
/dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /tmp
qa-ops02.mysite.com
/dev/mapper/sys-home   58G   26G   30G  47% /usr
/dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /var
qa-ops03.mysite.com
/dev/mapper/sys-home   58G   26G   30G  47% /lib
/dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /etc

to this one:

qa-ops01.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /home
qa-ops01.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /tmp  
qa-ops02.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /usr
qa-ops02.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /var   
qa-ops03.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /lib
qa-ops03.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /etc

I have used

cat FILE |sed 'N;s/.com\n//'

Is there anyway to achieve this, or should I just write the If... Then...

Thanks everybody for the answers :D (you always show me new things :D)


回答1:


The answer by potong is almost correct; it only handles one server, rather than multiple servers, but the change required is small.

$ sed -e '/^[^ ]*$/{h;d;}' -e 'G; s/\(.*\)\n\(.*\)/\2 \1/' data
qa-ops01.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /home
qa-ops01.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /tmp
qa-ops02.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /usr
qa-ops02.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /var
qa-ops03.mysite.com /dev/mapper/sys-home   58G   26G   30G  47% /lib
qa-ops03.mysite.com /dev/mapper/sys-tmp   3.9G  2.3G  1.5G  61% /etc
$

The script is in two parts, identified by the two -e options. The first part identifies server names; those lines contain no spaces (hence /^[^ ]*$/ looks for a line with no spaces), and copies the line into the hold space (h) and then deletes it (d) and continues with the next line. The second part of the script is only exercised on lines that contain spaces. It appends the content of the hold space to the pattern space after a newline (G); then it splits the line into 'everything up to the newline' and 'everything after the newline', and switches them so that the 'after' (\2) comes first, then a space, then the 'before' (\1).

This uses the classic sed regular expressions; it was tested on Mac OS X (10.7.5) with both the BSD sed and also with GNU sed without change. GNU sed has options such as -r to change the interpretation of regexes which would save you a few backslashes.




回答2:


I don't know much about sed, but in AWK:

awk 'NR==1{prefix=$0;next} {print prefix, $0}' file

Update

If that is the case, look for lines with only one field (column), use it for prefix. That means, the only change to the above script is NF (number of fields) in place of NR (number of records, or lines).

awk 'NF==1{prefix=$0;next} {print prefix, $0}' file



回答3:


This might work for you (GNU sed):

sed -r '1{h;d};G;s/(.*)\n(.*)/\2 \1/' file
  • 1{h;d} save the first line in the hold space (HS) and then delete the pattern space (PS).
  • G on all subsequent lines append a newline and the HS to PS.
  • s/(.*)\n(.*)/\2 \1/ re-arrange the PS and remove the introduced newline.

Since posting the answer the original question was changed, to take this into account:

sed -r '/%/!{h;d};G;s/(.*)\n(.*)/\2 \1/' file

This handles multiple servers by saving lines in the HS which do not contain %.




回答4:


try this one:

awk '!/^\/dev/{header=$0;next} {print header, $0}' input.txt

alternatively (functionally identical, but a bit easier to read and understand, IMO):

awk '{if($0 !~ /^\/dev/){header=$0}else{print header, $0}}' input.txt



回答5:


awk '{if($0~/mysite.com/){x=$0}else{print x,$0}}' your_file

tested here




回答6:


Another way using awk is as follows:

awk '{if ($1 ~ /mysite.com/){a= $0 } if ($1 ~ /dev/){ print a" "$0}}' temp.txt

  1. a = $0 copies the line to a if that contains mysite.com
  2. If second lines contains /dev then it joins both lines

or other variation of potong sed

sed -re '/mysite.com/{h;d};G;s/(.*)\n(.*)/\2 \1/' temp.txt



来源:https://stackoverflow.com/questions/14483341/bash-to-merge-lines-in-a-file

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!