Сопоставить строку и напечатать строки из одного блока, совпадающие с другой строкой
У меня есть большой файл с более чем 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
"всегда сопровождается":
"без места.