Adição de valores nas segundas colunas se a entrada da primeira coluna for a mesma no UNIX

Adição de valores nas segundas colunas se a entrada da primeira coluna for a mesma no UNIX

Estou tentando agregar um arquivo contendo os seguintes dados no UNIX. Preciso somar os valores se a chave for a mesma.

Key,amount,date,Time
abc-xyz-12234,45,15-08-91,23:00
pqr-vgh-5241,15,15-08-91,21:00
abc-xyz-12234,35,15-08-91,23:00
pqr-vgh-5241,24,15-08-91,21:00
abc-xyz-12234,655,15-08-91,23:00
lkj-erf-8542,281,15-08-91,10:00
pqr-vgh-5241,40,15-08-91,21:00

A saída deve ser a seguinte

abc-xyz-12234,735,15-08-91,23:00
pqr-vgh-5241,79,15-08-91,21:00
lkj-erf-8542,281,15-08-91,10:00

Eu tentei pelo seguinte comando, mas só me dá uniq

cat file | grep "abc-xyz-12234" | uniq

Responder1

Outra solução possível awkpoderia ser:

awk 'BEGIN { FS = OFS = "," }
     NR != 1 { y[$1] += $2; $2 = y[$1]; x[$1] = $0; }
     END { for (i in x) { print x[i]; } }' file

A falha é que não preservará seu pedido. Então o resultado poderia ser:

pqr-vgh-5241,79,15-08-91,21:00
abc-xyz-12234,735,15-08-91,23:00
lkj-erf-8542,281,15-08-91,10:00

Responder2

Você pode fazer isso com o awk:

#!/bin/sh
sort | awk -F, '
function result() {
    if ( key != "" ) {
            printf "%s,%d,%s\n", key, value, datetime;
    }
}
BEGIN { key = ""; value = 0; datetime = ""; }
$2 ~ /^[0-9]+/ {
    if ( $1 == key ) {
            value += $2;
    } else {
            result();
            key = $1;
            value = $2;
            datetime = $3 "," $4;
    }
}
END { result(); }
'

dando

./foo <input
abc-xyz-12234,735,15-08-91,23:00
lkj-erf-8542,281,15-08-91,10:00
pqr-vgh-5241,79,15-08-91,21:00

Responder3

Aqui está uma maneira em Perl. Chame-o assim ./script file.ext:

use warnings;
use strict;

my %data;
my @order;

while (<>){
    next if $. == 1;
    my @line = split /,/;
    if (defined $data{$line[0]}){
        $data{$line[0]}->[1] += $line[1];
    }
    else {
        $data{$line[0]} = \@line;
        push @order, $line[0];
    }
}

for (@order){
    print join(',', @{$data{$_}});
}

Responder4

Isso usa Miller ( mlr) para calcular a soma do amountcampo sobre cada conjunto de registros agrupados pelos campos Key, datee Time. Miller adicionará a soma como um novo campo, amount_sum, no final da lista de campos, para que também reordenemos os campos e os renomeemos.

mlr --csv \
    stats1 -a sum -f amount -g Key,date,Time then \
    reorder -f Key,amount_sum then \
    label Key,amount file

Saída considerando os dados da pergunta:

Key,amount,date,Time
abc-xyz-12234,735,15-08-91,23:00
pqr-vgh-5241,79,15-08-91,21:00
lkj-erf-8542,281,15-08-91,10:00

Se você não quiser o cabeçalho na saída, use mlrcom sua --headerless-csv-outputopção.

informação relacionada