Adição com 'sed'

40

Estou tentando executar uma operação matemática com sed, mas continua a tratar minhas variáveis ​​como seqüências de caracteres. A entrada é deste tipo:

$ echo 12 | sed 's/[0-9]*/&+3/'
$ 12+3

Eu gostaria de ter 15 como saída. Preciso fazer a operação e substituir seu resultado matemático em apenas uma passagem, porque estou executando o programa como um daemon Python e quero evitar passagens como redirecionar stdoutarquivos, abrir esses arquivos, executar operações, extrair o resultado, fazer a substituição. Para mim, sedparece o melhor para executar tudo em uma linha.

Eu tentei converter a entrada e a saída de várias maneiras, como

$ echo 12 | sed 's/[0-9]*/int(&+3)/'
$ echo 12 | sed 's/[0-9]*/\int(&+3)/'
$ echo 12 | sed 's/[0-9]*/\int(&+3)/'

mas o resultado sempre foi a impressão do segundo campo.

Luigi Tiburzi
fonte
12
É tratar suas "variáveis" como strings, porque é tudo o que o sed faz - manipulação de strings. Não tem conceito de "número inteiro".
21412 Kevin
2
Estou muito curioso por que você deseja usar sedpara fazer matemática
David Oneill
Eu apenas pensei que poderia facilmente converter variáveis, não percebi que era tão complexo!
Luigi Tiburzi

Respostas:

82

Se você honestamente deseja usar o sed, este é o caminho a seguir:

s/[0-9]/<&/g
s/0//g; s/1/|/g; s/2/||/g; s/3/|||/g; s/4/||||/g; s/5/|||||/g; s/6/||||||/g
s/7/|||||||/g; s/8/||||||||/g; s/9/|||||||||/g
: tens
s/|</<||||||||||/g
t tens
s/<//g
s/+//g
: minus
s/|-|/-/g
t minus
s/-$//
: back
s/||||||||||/</g
s/<\([0-9]*\)$/<0\1/
s/|||||||||/9/; s/||||||||/8/; s/|||||||/7/; s/||||||/6/; s/|||||/5/; s/||||/4/
s/|||/3/; s/||/2/; s/|/1/
s/</|/g
t back

Entrada:

1+2
100+250
100-250

Saída:

3
350
-150

Sua missão, se você optar por aceitá-la, é implementar a multiplicação.

Simon Richter
fonte
5
+1 para o desafio, adorei! Talvez isso seria algo para Código Golf ;-p
Tatjana Heuser
6
E algumas pessoas dizem que programar não é matemática. Esta pequena jóia refuta todos eles. Melhor uso da Base 1 de todos os tempos.
precisa
11
Agradável! - @Simon: Eu desafio você a implementar tetração : P
AT
16
+1 Este é um belo exemplo do que um equívoco associado à criatividade pode gerar.
rozcietrzewiacz
11
A multiplicação é feita com o sed - e também aumenta muito bem!
Toby Speight
20

sednão é a melhor opção aqui, não faz aritmética de forma nativa (consulte Incrementar um número para saber como você poderia fazê-lo). Você poderia fazer isso com awk:

$ echo 12 | awk '{print $0+3}'
15

A melhor parte do código a ser usada dependerá do formato exato da sua entrada e do que você deseja / precisa fazer se não for numérico ou se contiver mais de um número, etc.

Você também pode fazer isso apenas com bash:

$ echo $(( $(echo 12) + 3 ))

ou usando de exprmaneira semelhante.

Esteira
fonte
17

Tentei aceitar o seu desafio @Richter, foi o que fiz usando parte do seu código:

sed 's/[0-9]/<&/g
s/0//g; s/1/|/g; s/2/||/g; s/3/|||/g; s/4/||||/g; s/5/|||||/g; s/6/||||||/g
s/7/|||||||/g; s/8/||||||||/g; s/9/|||||||||/g
: tens
s/|</<||||||||||/g
t tens
s/<//g
s/.*\*$/0/
s/^\*.*/0/
s/*|/*/
: mult
s/\(|*\)\*|/\1<\1*/ 
t mult
s/*//g
s/<//g
: back
s/||||||||||/</g
s/<\([0-9]*\)$/<0\1/
s/|||||||||/9/; s/||||||||/8/; s/|||||||/7/; s/||||||/6/; s/|||||/5/; s/||||/4/
s/|||/3/; s/||/2/; s/|/1/
s/</|/g
t back'

Entrada:

04*3
4*3
40*3
42*32
150*20
1*3
3*1
0*3
3*0

Saída: todos os resultados corretos

Luigi Tiburzi
fonte
@SimonRichter espero que você goste !!
Luigi Tiburzi
Cross postou esta resposta brilhante aqui: codegolf.stackexchange.com/a/39882/11259
Digital Trauma
12

perlpermite uma construção muito semelhante à sed... uma diferença é que perlpode fazer coisas mais complexas ... sedé muito bom para subestações de texto simples

 echo 'a12' | perl -pe 's/([0-9]+)/($1+3)/e'  # the trailing /e means evaluate

saída

a15
Peter.O
fonte
2
também pode fazer isso sem os parênteses de captura:perl -pe 's/[0-9]+/$&+3/e'
Glenn Jackman
8

basta alimentar a corda em uma calculadora

 echo 12 | sed 's/[0-9]*/&+3/' | bc
Glenn Jackman
fonte
isso não funcionará se houver texto entre os números.
Glenn Jackman
6

Realmente não entendo por que a extrema complexidade da resposta aceita, uma das opções abaixo, faz o que você deseja:

echo 12 | sed 's/[0-9]*/echo \$(( & + 3 ))/e'

ou

echo 12 | sed 's/[0-9]*/expr & + 3/e'

Eu acho que pode exigir o GNU sed, mas não tenho certeza.

michelpm
fonte
É uma extensão gnu.
30713 Kevin
Ok você está certo, mas a resposta vai além, ele implementa a adição geral, não um particular, você pode alimentar quaisquer dois números, e você obterá o resultado
Luigi Tiburzi
@LuigiTiburzi É bastante simples generalizar isso para entrada no estilo "x + y":echo 12+3 | sed -r 's/([0-9]*) *\+ *([0-9]*)/expr \1 + \2/e'
Digital Trauma
5

Se você definitivamente precisar combinar expressões regulares e operações aritméticas, escolha um idioma em que o parâmetro de substituição da expressão regular possa ser uma função de retorno de chamada.

Perl, Ruby, JavaScript e Python são essas linguagens:

bash-4.2$ echo 12 | perl -pe 's/\d+/$&+3/e'
15

bash-4.2$ echo 12 | ruby -pe '$_.sub!(/\d+/){|s|s.to_i+3}'
15

bash-4.2$ echo 12 | js -e 'print(readline().replace(/\d+/,function(s){return parseInt(s)+3}))'
15

bash-4.2$ echo 12 | python -c 'import re;print re.sub("\d+",lambda s:str(int(s.group(0))+3),raw_input())'
15
homem a trabalhar
fonte
1

Outra bashsolução simples , que realmente funciona em um pipe:

 echo 12 | { read num; echo $(( num + 3)); }
rozcietrzewiacz
fonte
1

Se você misturar algum basismo:

echo $(($(echo 12 | sed 's/[0-9]*/&+3/')))

Para extrair o número de um texto:

echo $(($(echo "foo12bar" | sed -r 's/[^0-9]*([0-9]*).*/\1+3/')))

Sem sed, basta bash:

var="foo12bar"
echo $((${var//[^0-9]/}+3))

substitui todos os que não são dígitos ${var//[^0-9]/}e faz aritmética em parênteses de dupla rodada:$((x+3))

Usuário desconhecido
fonte
2
Não há basismo aqui. $((...))foi introduzido pelo POSIX (o basismo é $[...]). ${var//xxx/x}é um kshism também copiado pelo zsh e bash. sed -ré um GNUism
Stéphane Chazelas
0

Aqui está uma solução Perl:

echo 12 | perl -wlpe '$_ += 3'
# Output:  15

Se você preferir alterar o primeiro conjunto de dígitos encontrado em uma sequência, poderá usar:

echo I am 12 years old. | perl -wlpe 's/(\d+)/$1 + 3/e'
# Output:  I am 15 years old.

Se você preferir alterar todos os conjuntos de dígitos em uma string, poderá usar o /gmodificador, assim:

echo They are 11, 12, and 13 years old. | perl -wlpe 's/(\d+)/$1 + 3/eg'
# Output:  They are 14, 15, and 16 years old.
JL
fonte
0

Embora o uso da expressão sed seja ótimo, ele tem suas limitações. Por exemplo, o seguinte falha:

$ echo "1000000000000000000000000000000+1" | sed -e 's/\([0-9]*\)+\([0-9]*\)/expr \1 + \2/e'
expr: 1000000000000000000000000000000: Numerical result out of range

Para superar essa limitação, basta recorrer ao poder interno do puro sed e implementar o seguinte somador decimal de comprimento arbitrário:

#! / bin / sed -f

s / + / \ n / g
s / $ / \ n \ n0 /

: LOOP
s / ^ \ (. * \) \ (. \) \ n \ (. * \) \ (. \) \ n \ (. * \) \ n \ (. \) $ / 0 \ 1 \ n0 \ 3 \ n \ 5 \ n \ 6 \ 2 \ 4 /
h
s /^.* \ n. * \ n. * \ n \ (... \) $ / \ 1 /

# módulo somador decimal completo
# INPUT: 3digits (Entrada, A, B,)
# SAÍDA: 2 bits (Carry, Sum)
s / $ /;000 = 00001 = 01002 = 02003 = 03004 = 04005 = 05006 = 06007 = 07008 = 08009 = 09010 = 01011 = 02012 = 03013 = 04013 = 04014 = 05015 = 06016 = 07017 = 08018 = 09019 = 10020 = 02021 = 03022 = 04023 = 05024 = 06025 = 07026 = 08027 = 09028 = 10029 = 11030 = 03031 = 04032 = 05033 = 06034 = 07035 = 08036 = 09037 = 10038 = 11039 = 12040 = 04041 = 05042 = 06043 = 07044 = 08045 = 09046 = 10047 = 11048 = 13050 = 05051 = 06052 = 07053 = 08054 = 09055 = 10056 = 11057 = 12058 = 13059 = 14060 = 06061 = 07062 = 08063 = 09064 = 10090 = 11066 = 12067 = 13068 = 14069 = 15070 = 07071 = 08072 = 09073 = 10074 = 11075 = 12076 = 13077 = 14078 = 15079 = 16080 = 08081 = 09082 = 10083 = 11084 = 12085 = 13086 = 14087 = 15088 = 16089 = 17090 = 09091 = 10092 = 11093 = 12094 = 13095 = 14096 = 15097 = 16098 = 17099 = 18100 = 01101 = 02102 = 03103 = 04104 = 05105 = 06106 = 07107 = 08108 = 09109 = 10110 = 02111 = 03112 = 04113 = 05114 = 06115 = 07116 = 08117 = 09118 = 10119 = 11120 = 03121 = 04122 = 05123 = 06124 07125 = 08126 = 09127 = 10128 = 11129 = 12130 = 04131 = 05132 = 06133 = 07134 = 08135 = 09136 = 10137 = 11138 = 12139 = 13140 = 05141 = 06142 = 07143 = 08144 = 09145 = 10146 = 11147 = 12148 = 13149 = 14150 = 06151 = 07152 = 08153 = 09154 = 10155 = 11156 = 12157 = 13158 = 14159 = 15160 = 07161 = 08162 = 09163 = 10164 = 11165 = 13167 = 14168 = 15169 = 16170 = 08171 = 09172 = 10173 = 11174 = 12175 = 13176 = 14177 = 15178 = 16179 = 17180 = 09181 = 10182 = 11183 = 12184 = 13185 = 14186 = 15187 = 16188 = 17189 = 18192 = 10191 = 11174 12193 = 13194 = 14195 = 15196 = 16197 = 17198 = 18199 = 19 /
s / ^ \ (... \) [^;] *; [^;] * \ 1 = \ (.. \). * / \ 2 /
H
g
s / ^ \ (. * \) \ n \ (. * \) \ n \ (. * \) \ n ... \ n \ (. \) \ (. \) $ / \ 1 \ n \ 2 \ n \ 5 \ 3 \ n \ 4 /
/ ^ \ ([0] * \) \ n \ ([0] * \) \ n / {
        s /^.* \ n. * \ n \ (. * \) \ n \ (. \) / \ 2 \ 1 /
        s / ^ 0 \ (. * \) / \ 1 /
        q
}
b LOOP

A maneira como funciona é implementando o módulo somador decimal que adiciona dois dígitos de entrada (A e B), bem como Carry Bit e produz um bit Sum e Carry. A idéia é emprestada da eletrônica, onde o somador binário faz o mesmo para números binários. Tudo o que precisamos fazer é passar o somador por todos os dígitos e podemos adicionar números arbitrários de comprimento (limitados pela memória). Abaixo está o adicionador em ação:

./decAdder.sed
666666666666666666666666666666999999999999991111111112222+1100000000000000000000011111111111111111111111111111111111
1766666666666666666666677777778111111111111102222222223333

Da mesma maneira, pode-se implementar o somador binário (ou qualquer outra base). Tudo o que você precisa fazer é substituir a linha que começa s/$/;000=00001...com o padrão de substituição apropriado para a base especificada. Por exemplo: s/$/;000=00001=01010=01011=10100=01101=10110=10111=11/ é o padrão de substituição do somador binário de comprimento arbitrário.

Você pode ajustar o código documentado no meu github .

Emsi
fonte