Прочтите файл построчно, и если условие соблюдается, продолжайте чтение до следующего условия [закрыто]

echo "${value/bigint_col/my_col}"  # outputs contents of $value replacing the first instance of 'bigint_col' with 'my_col'
4
19.01.2016, 01:56
1 ответ

Вам необходимо внести некоторые изменения в ваш скрипт (без особого порядка):

  • Используйте IFS= перед read, чтобы избежать удаления ведущих и корявых пробелов.
  • Поскольку $line нигде не изменяется, нет необходимости в переменной readLine.
  • Не используйте read в середине цикла!!!
  • Используйте булеву переменную для управления печатью.
  • Четко определите начало и конец печати.

С этими изменениями скрипт становится:

#!/bin/bash

filename="foo.txt"

#While loop to read line by line
while IFS= read -r line; do
    #If the line starts with ST then set var to yes.
    if [[ $line == qwe* ]] ; then
        printline="yes"
        # Just t make each line start very clear, remove in use.
        echo "----------------------->>"
    fi
    # If variable is yes, print the line.
    if [[ $printline == "yes" ]] ; then
        echo "$line"
    fi
    #If the line starts with ST then set var to no.
    if [[ $line == ewq* ]] ; then
        printline="no"
        # Just to make each line end very clear, remove in use.
        echo "----------------------------<<"
    fi
done < "$filename"

Который можно сократить так:

#!/bin/bash
filename="foo.txt"
while IFS= read -r line; do
    [[ $line == qwe* ]]       && printline="yes"
    [[ $printline == "yes" ]] && echo "$line"
    [[ $line == ewq* ]]       && printline="no"
done < "$filename"

Который будет печатать начальную и конечную строки (включительно).
Если нет необходимости печатать их, поменяйте местами начальный и конечный тесты:

#!/bin/bash
filename="foo.txt"
while IFS= read -r line; do
    [[ $line == ewq* ]]       && printline="no"
    [[ $printline == "yes" ]] && echo "$line"
    [[ $line == qwe* ]]       && printline="yes"
done < "$filename"

Однако, лучше (если у вас bash версии 4.0 или лучше) использовать readarray и выполнить цикл с элементами массива:

#!/bin/dash
filename="infile"

readarray -t lines < "$filename"


for line in "${lines[@]}"; do
    [[ $line == ewq* ]]       && printline="no"
    [[ $printline == "yes" ]] && echo "$line"
    [[ $line == qwe* ]]       && printline="yes"
done

Это позволит избежать большинства проблем, связанных с использованием read.


Конечно, вы можете использовать рекомендуемую (в комментариях; спасибо, @costas) строку sed, чтобы получить только строки для обработки:

    #!/bin/bash
filename="foo.txt"

readarray -t lines <<< "$(sed -n '/^qwe.*/,/^ewq.*/p' "$filename")"

for line in "${lines[@]}"; do

     : # Do all your additional processing here, with a clean input.

done 
7
27.01.2020, 20:47

Теги

Похожие вопросы