Apagar n linhas depois do padrão e m linhas antes do padrão

2

NOTA : Estou ciente desta questão: Como grep-inverse-match e exclua as linhas" before "e" after ". O meu não é uma duplicata como havia sido anteriormente citado, as respostas para essa questão também apagam o padrão, enquanto que nesta questão o próprio padrão deve ser preservado.

Estou tentando apagar n linhas depois de um padrão e m linhas antes de um padrão (sem excluir o padrão). Por exemplo, se o arquivo for:

1
2
3
4
5
6
7
8
9

Se, Padrão = 5 e n = 2 e m = 3. Então:

1
5
8
9

Você poderia, por favor, sugerir como fazer isso?

Bônus : seria ótimo se pudéssemos definir m ou n = 0 no mesmo código. Por exemplo. Se definirmos m = 0 e n = 1, no exemplo acima, devemos obter:

1
2
3
4
5
7
8
9
    
por Nikhil 06.10.2018 / 12:31

2 respostas

3

Para responder à sua situação geral, construímos um código ed apropriado com antecedência, com base no tipo de dados fornecidos.

 re=5 n=2 m=3
 code=$(
   prev="/$re/-$m,/$re/-1d"
   next="/$re/+1,/$re/+${n}d"
    case "$m/$n" in
       0/0) set -- ;;
       0/?*) set -- "$next" "w" ;;
       ?*/0) set -- "$prev" "w" ;;
         *) set -- "$prev" "$next" "w" ;;
    esac
    printf '%s\n' ${1+"$@"} "q" 
 ) 
 ed -s filename - <<eof
 $code
 eof

Uma maneira pode ser: isso usa o editor ed para realizar o endereçamento relativo, pois é nisso que o seu problema está centrado.

 n=3 m=2 re=5
 ed -s filename - <<eof
 /$re/-$m,/$re/-1d
 .+1,.+${n}d
 wq
 eof

Explicação:

 1. Line 3 after var substitution becomes
            /5/-2,/5/-1
      What it does is, sitting on line which satisfies the regex /5/, it looks 2 lines behind and stops looking 1 line before the /5/ line or the current line and deletes that bunch. Remember the current line is not deleted.

   2.  Line 4, after var sub becomes
                .+1,.+3d
       . is the nickname for the current line, which in our case is /5/
       So, starting fron one line after the current upto 3 lines after the current, delete them all. Note the current line is still untouched.

  3. Line 5 is wq which means save the modified file back onto itself and quit.

For more on info google the manual for gnu ed editor. 
    
por 06.10.2018 / 16:56
2

Digamos que você tenha um arquivo - numbers.list :

1
2
3
4
5
6
7
8
9

Aqui está um script que deve realizar o que você procura:

#!/bin/bash
if [ -z "$1" ]; then
    echo >&2 "error: no file specified"
    exit 1
fi
if [ ! -f "$1" ]; then
    echo >&2 "error: $1 is not a file"
    exit 1
fi
if [ -z "$2" ]; then
    echo >&2 "error: no pattern specified"
    exit 1
fi
grep "$2" "$1" >/dev/null 2>&1
if [ ! 0 -eq "$?" ]; then
    echo >&2 "error: pattern $2 not found in $1"
    exit 1
fi
MATCH_FILE="$1"
MATCH_PATTERN="$2"
MATCH_DELETE_AFTER=0
MATCH_DELETE_BEFORE=0
if [ ! -z "$3" ]; then
    MATCH_DELETE_AFTER="$3"
fi
if [ ! -z "$4" ]; then
    MATCH_DELETE_BEFORE="$4"
fi
MATCH_FILE_LINE="$( grep -n "$2" "$1" | cut -d: -f1 )"
#
# print matching 'head' minus MATCH_DELETE_BEFORE lines
cat "$1" \
    | head -n "$( expr "$MATCH_FILE_LINE" - "$MATCH_DELETE_BEFORE" - 1 )"
#
# print matching line
cat "$1" \
    | head -n "$MATCH_FILE_LINE" \
    | tail -n 1
#
# print matching 'tail' minus MATCH_DELETE_AFTER lines
cat "$1" \
    | tail -n "$( expr "$( wc -l "$1" | cut -d' ' -f1 )" - "$MATCH_FILE_LINE" - "$MATCH_DELETE_AFTER" )"

Exemplo de uso: ./matching.sh numbers.list 5 2 3

Observação: essa solução não funcionará como esperado se expr avaliar valores negativos - até você se quiser implementar verificações para evitar esse comportamento.

    
por 06.10.2018 / 12:58