it-swarm-ko.tech

sed를 사용하여 여러 공백을 하나로 제거하는 방법은 무엇입니까?

AIX의 sed는 내가 생각하는 바를 수행하지 않습니다. IOSTAT 출력에서 ​​여러 공백을 단일 공백으로 바꾸려고합니다.

# iostat
System configuration: lcpu=4 drives=8 paths=2 vdisks=0

tty:      tin         tout    avg-cpu: % user % sys % idle % iowait
          0.2         31.8                9.7   4.9   82.9      2.5

Disks:        % tm_act     Kbps      tps    Kb_read   Kb_wrtn
hdisk9           0.2      54.2       1.1   1073456960  436765896
hdisk7           0.2      54.1       1.1   1070600212  435678280
hdisk8           0.0       0.0       0.0          0         0
hdisk6           0.0       0.0       0.0          0         0
hdisk1           0.1       6.3       0.5   63344916  112429672
hdisk0           0.1       5.0       0.2   40967838  98574444
cd0              0.0       0.0       0.0          0         0
hdiskpower1      0.2     108.3       2.3   2144057172  872444176

# iostat | grep hdisk1
hdisk1           0.1       6.3       0.5   63345700  112431123

#iostat|grep "hdisk1"|sed -e"s/[ ]*/ /g"
 h d i s k 1 0 . 1 6 . 3 0 . 5 6 3 3 4 5 8 8 0 1 1 2 4 3 2 3 5 4

sed는 전체 그룹 (/ g)에 대해 여러 공백 (/ [] * /)을 단일 공백 ​​(/ /)으로 검색하고 대체해야하지만 ... 각 문자 사이의 간격을 유지해야합니다.

내가 무엇을 잘못하고 있지? 나는 그것이 간단한 것이어야한다는 것을 안다 ... AIX 5300-06

edit : 10 개 이상의 하드 드라이브가있는 다른 컴퓨터가 있습니다. 이것을 모니터링 목적으로 다른 프로그램의 매개 변수로 사용하고 있습니다.

내가 부딪친 문제는 2 단계에서 $ 1 등을 사용하고 있고 인쇄 명령에 오류가있어서 "awk '{print $ 5}'가 작동하지 않는다는 것입니다. 나는 grep/sed/cut 버전을 찾고있었습니다 작동하는 것 같습니다 :

iostat | grep "hdisk1 " | sed -e's/  */ /g' | cut -d" " -f 5

그들이 "단 하나"를 의미한다고 생각했을 때 []는 "0 이상"이었습니다. 브래킷을 제거하면 작동합니다. 세 가지 좋은 답변은 실제로 "답변"을 선택하기 어렵게 만듭니다.

72
WernerCD

grep의 사용은 중복 적이며 sed도 동일하게 수행 할 수 있습니다. 문제는 공백도 0 인 *를 사용하는 것입니다. 대신 \+를 사용해야합니다.

iostat | sed -n '/hdisk1/s/ \+/ /gp'

sed\+ 메타 문자를 지원하지 않으면

iostat | sed -n '/hdisk1/s/  */ /gp'
57
enzotib

/[ ]*/zero 이상의 공백과 일치하므로 문자 사이의 빈 문자열이 일치합니다.

"하나 이상의 공백"과 일치 시키려면 다음 중 하나를 사용하십시오.

... | sed 's/  */ /g'
... | sed 's/ \{1,\}/ /g'
... | tr -s ' '
75
glenn jackman

* 연산자를 +. 공백이 아닌 모든 것이 ... 음 ... 제로 인스턴스이기 때문에 모든 문자와 일치하는 이전 문자 중 0 이상을 일치시킵니다. 하나 이상 일치해야합니다. 실제로 두 개 이상 일치하는 것이 좋습니다

한 문자를 일치시키는 데 대괄호로 묶은 문자 클래스도 필요하지 않습니다. 당신은 사용할 수 있습니다 :

s/  \+/ /g

... 탭이나 다른 종류의 공백과 일치시키지 않으려면 문자 클래스를 사용하는 것이 좋습니다.

15
Caleb

다음과 같은 순서로 마지막 항목을 항상 일치시킬 수 있습니다.

s/\(sequence\)*/\1/

그래서 당신은 올바른 길을 가고 있지만 시퀀스를 공백으로 바꾸는 것이 아니라 마지막 공백 인 단일 공백으로 바꾸십시오. 이렇게하면 is 공간 시퀀스가 ​​일치하면 시퀀스가 ​​단일 공간으로 축소되지만 null 문자열이 일치하면 null 문자열이 자체로 바뀌고 아무런 해가 없으며 파울이 없습니다. 예를 들어,

sed 's/\( \)*/\1/g' <<\IN                                    
# iostat
System configuration: lcpu=4 drives=8 paths=2 vdisks=0

tty:      tin         tout    avg-cpu: % user % sys % idle % iowait
          0.2         31.8                9.7   4.9   82.9      2.5

Disks:        % tm_act     Kbps      tps    Kb_read   Kb_wrtn
hdisk9           0.2      54.2       1.1   1073456960  436765896
hdisk7           0.2      54.1       1.1   1070600212  435678280
hdisk8           0.0       0.0       0.0          0         0
hdisk6           0.0       0.0       0.0          0         0
hdisk1           0.1       6.3       0.5   63344916  112429672
hdisk0           0.1       5.0       0.2   40967838  98574444
cd0              0.0       0.0       0.0          0         0
hdiskpower1      0.2     108.3       2.3   2144057172  872444176

# iostat | grep hdisk1
hdisk1           0.1       6.3       0.5   63345700  112431123

IN

산출

# iostat
System configuration: lcpu=4 drives=8 paths=2 vdisks=0

tty: tin tout avg-cpu: % user % sys % idle % iowait
 0.2 31.8 9.7 4.9 82.9 2.5

Disks: % tm_act Kbps tps Kb_read Kb_wrtn
hdisk9 0.2 54.2 1.1 1073456960 436765896
hdisk7 0.2 54.1 1.1 1070600212 435678280
hdisk8 0.0 0.0 0.0 0 0
hdisk6 0.0 0.0 0.0 0 0
hdisk1 0.1 6.3 0.5 63344916 112429672
hdisk0 0.1 5.0 0.2 40967838 98574444
cd0 0.0 0.0 0.0 0 0
hdiskpower1 0.2 108.3 2.3 2144057172 872444176

# iostat | grep hdisk1
hdisk1 0.1 6.3 0.5 63345700 112431123

그러나이 상황에서 정규 표현식을 완전히 피하고 대신 수행하는 것이 훨씬 좋습니다.

tr -s \  <infile
8
mikeserv

시도하는 작업을 수행 할 수도 있습니다.

iostat | grep "hdisk1 " | sed -e's/  */ /g' | cut -d" " -f 5

으로

iostat | while read disk tma kbps tps re wr; do [ "$disk" = "hdisk1" ] && echo "$re"; done

나중에 다른 필드에 액세스하거나 무언가를 계산하려고 할 때 특히 유용 할 수 있습니다.

iostat | while read disk tma kbps tps re wr; do [ "$disk" = "hdisk1" ] && echo "$(( re/1024 )) Mb"; done
5
rozcietrzewiacz

다음 스크립트를 사용하여 여러 공백을 단일 공백, TAB 또는 다른 문자열로 변환 할 수 있습니다.

$ ls | compress_spaces.sh       # converts multiple spaces to one
$ ls | compress_spaces.sh TAB   # converts multiple spaces to a single tab character
$ ls | compress_spaces.sh TEST  # converts multiple spaces to the phrase TEST
$ compress_spaces.sh help       # show the help for this command

compress_spaces.sh

function show_help()
{
  IT=$(CAT <<EOF

  usage: {REPLACE_WITH}

  NOTE: If you pass in TAB, then multiple spaces are replaced with a TAB character

  no args -> multiple spaces replaced with a single space
  TAB     -> multiple spaces replaced with a single tab character
  TEST    -> multiple spaces replaced with the phrase "TEST"

  )
  echo "$IT"
  exit
}

if [ "$1" == "help" ]
then
  show_help
fi

# Show help if we're not getting data from stdin
if [ -t 0 ]; then
  show_help
fi

REPLACE_WITH=${1:-' '}

if [ "$REPLACE_WITH" == "tab" ]
then
  REPLACE_WITH=$'\t'
fi
if [ "$REPLACE_WITH" == "TAB" ]
then
  REPLACE_WITH=$'\t'
fi

sed "s/ \{1,\}/$REPLACE_WITH/gp"
0
Brad Parks