วิธีนับแถวที่เรียงลำดับโดยเขตข้อมูลแรกใน bash


9

นี่คือตัวอย่างจาก INPUT:

...
####################
Bala Bela;XXXXXX12345;XXXXXX12345678;A
SERVER345Z3.DOMAIN.com0
SERVER346Z3.DOMAIN.com0
SERVER347Z3.DOMAIN.com0
SERVER348Z3.DOMAIN.com0
ssh-dss ...pubkeyhere...
####################
Ize Jova;XXXXXX12345;XXXXXX12345;A
SERVER342Z3.DOMAIN.com0
SERVER343Z3.DOMAIN.com0
SERVER345Z3.DOMAIN.com0
ssh-rsa ...pubkeyhere...
...

และนี่คือตัวอย่างจากผลลัพธ์ที่ฉันต้องการ:

Bala Bela;XXXXXX12345;XXXXXX12345678;A
4
Ize Jova;XXXXXX12345;XXXXXX12345;A
3

ดังนั้นฉันต้องการ OUTPUT จาก INPUT เพื่อที่ฉันจะได้เห็นว่ามีกี่แถวที่ขึ้นต้นด้วย "SERVER" ไปถึงผู้ใช้ที่ให้ไว้ (เช่น: "Bala Bela; XXXXXX12345; XXXXXX12345678; A") ฉันจะทำสิ่งนี้ในทุบตีได้อย่างไร


คุณต้องการให้สิ่งนี้เป็น Bash แบบสแตนด์อโลนหรือเป็นเครื่องมืออื่นที่ยอมรับได้ (grep, awk, perl ... )?
ire_and_curses

ฉันจะถือว่า (และได้ทำเช่นนั้น :) ว่าเว้นแต่ระบุไว้อย่างชัดเจนคำถามทุบตีสคริปต์ช่วยให้เครื่องมือมาตรฐานทั้งหมดเช่น grep, awk, sed, perl และส่วนที่เหลือทั้งหมด
cas

คำตอบ:


6
{
i=0
while IFS= read -r line; do
  case "$line" in
    ssh*|'##'*)
      ;;
    SERVER*)
      ((++i))
      ;;
    *)
      if ((i>0)); then echo $i;i=0; fi
      echo "$line"
      ;;
  esac
done
if ((i>0)); then echo $i;i=0; fi
} <inputfile >outputfile

เหมือนกันใน perl หนึ่งซับ

perl -nle '
  BEGIN{$i=0}
  next if/^(ssh|##)/;
  if(/^SERVER/){++$i;next}
  print$i if$i>0;
  $i=0;
  print;
  END{print$i if$i>0}' inputfile >outputfile

และ golfed

perl -nle's/^(ssh|##|(SERVER))/$2&&$i++/e&&next;$i&&print$i;$i=!print}{$i&&print$i' inputfile >outputfile

ว้าว. perl เป็นสิ่งที่น่าอัศจรรย์: D
gasko peter

5

รุ่นนี้นับแถวทั้งหมดที่ไม่ตรงกับ regexp ในgrepบรรทัด

#! /usr/bin/perl 

# set the Input Record Separator (man perlvar for details)
$/ = '####################';

while(<>) {
    # split the rows into an array
    my @rows = split "\n";

    # get rid of the elements we're not interested in
    @rows = grep {!/^#######|^ssh-|^$/} @rows;

    # first row of array is the title, and "scalar @rows"
    # is the number of entries, so subtract 1.
    if (scalar(@rows) gt 1) {
      print "$rows[0]\n", scalar @rows -1, "\n"
    }
}

เอาท์พุท:

Bala Bela; XXXXXX12345; XXXXXX12345678; A
4
Ize Jova; XXXXXX12345; XXXXXX12345; A
3

ถ้าคุณเพียงต้องการที่จะนับจำนวนบรรทัดที่เริ่มต้นด้วย SERVER 'แล้ว:

#! /usr/bin/perl 

# set the Input Record Separator (man perlvar for details)
$/ = '####################';

while(<>) {
    # split the rows into an array
    my @rows = split "\n";

    # $rows[0] will be same as $/ or '', so get title from $rows[1]
    my $title = $rows[1];

    my $count = grep { /^SERVER/} @rows;

    if ($count gt 0) {
      print "$title\n$count\n"
    }
}

5
sed -n ':a /^SERVER/{g;p;ba}; h' file | uniq -c | 
  sed -r 's/^ +([0-9]) (.*)/\2\n\1/'

เอาท์พุท:

Bala Bela;XXXXXX12345;XXXXXX12345678;A
4
Ize Jova;XXXXXX12345;XXXXXX12345;A
3

หากการนับคำนำหน้าเป็น ok:

sed -n ':a /^SERVER/{g;p;ba}; h' file |uniq -c

เอาท์พุท:

  4 Bala Bela;XXXXXX12345;XXXXXX12345678;A
  3 Ize Jova;XXXXXX12345;XXXXXX12345;A

4

awkทางเลือก:

/^#{15,}/ {           # if line starts with 15 or more number signs
  if(k) {             # if any key found
    print k RS n      # print it and occurrences of SERVER
    n=0
  }
  getline             # key is on the next line
  k = $0
  next                # move to next record
} 

/SERVER/ { n++ }      # count occurrences of SERVER
END { print k RS n }  # print last record

ทั้งหมดในหนึ่งบรรทัด:

awk '/^#{15,}/ { if(n>0) { print k RS n; n=0 }; getline; k = $0; next } /SERVER/ { n++ } END { print k RS n }'

2

ดังนั้นหากผลลัพธ์ถูกจัดเรียงแล้วใน "bucket" แต่ละรายการคุณสามารถใช้ uniq โดยตรงกับการตรวจสอบเฉพาะอักขระ N ตัวแรก:

cat x | uniq -c -w6

นี่คือ N == 6 เนื่องจาก SERVER ประกอบด้วยอักขระ 6 ตัวตั้งแต่ต้นบรรทัด คุณจะจบลงด้วยผลลัพธ์นี้ (ซึ่งแตกต่างจากผลลัพธ์ที่คุณต้องการ):

  1 ####################
  1 Bala Bela;XXXXXX12345;XXXXXX12345678;A
  4 SERVER345Z3.DOMAIN.com0
  1 ssh-dss ...pubkeyhere...
  1 ####################
  1 Ize Jova;XXXXXX12345;XXXXXX12345;A
  3 SERVER342Z3.DOMAIN.com0
  1 ssh-rsa ...pubkeyhere...
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.