Сопоставить строку и напечатать строки из одного блока, совпадающие с другой строкой

У меня есть большой файл с более чем 5000 строк в следующем формате

Ниже фрагмент показывает два блока файла.

string name    : abcd

    used :metric
    test :ok

{


 fun: add

 fun: sub

 fun: mul

 fun: div

}   


string name    : degh

    used: non -metric
    test: good

{


 fun: per

 fun: div

 fun: add

 fun: mul


}   

Что мне нужно, так это искать string name (например: abcd), а затем выведите значения после fun : От этого string nameблок

Я хотел бы следующий вывод:

abcd    add
abcd    sub
abcd    mul
abcd    div
degh    per
degh    div
degh    add
degh    mul

Каков был бы правильный способ решить эту проблему?

3 ответа

Один из способов подойти к нему с помощью Perl:

$ perl -lane '$hold=$F[3] if $_ =~ "^string name.*";print "$hold $F[1]" if $F[0] eq "fun:"' bigfile.txt                                                                
abcd add
abcd sub
abcd mul
abcd div
degh per
degh div
degh add
degh mul
#!/bin/bash

RE_NAME='^ *string name *:' # regex for the 'name' line
RE_FUNSTART='^ *[{] *$'  # regex for the start of the 'fun' block
RE_FUNEND='^ *[}] *$'  # regex for end of 'fun' block
RE_FUN='^ *fun:'  # regex for 'fun' line

while read line; do
 if [[ $line =~ $RE_NAME ]]; then
     name="${line##*: }"
     echo
 elif [[ $line =~ $RE_FUNSTART ]]; then
     fun='1'
 elif [[ $line =~ $RE_FUNEND ]]; then
     fun=''
 elif [[ ($line =~ $RE_FUN) && (-n $fun) ]];  then   # match 'fun' lines only inside 'fun' block
     echo "$name    ${line##*: }"
 fi

done < your_big_file

Bash может быть немного медленным для больших файлов. Если он слишком медленный для вас, вы можете перенести код, например, на Perl или Python.

Другой подход с awk:

 awk '{ if ($1 == "string") name = $4; else if ($1 == "fun:") print name " " $2; }' your_file

При условии, что "string name" а также ":"разделены пробелом и"fun"всегда сопровождается":"без места.

Другие вопросы по тегам