while: Unterschied zwischen den Versionen

Aus Mikiwiki
Zur Navigation springen Zur Suche springen
Zeile 90: Zeile 90:
   do
   do
     sleep 1 &
     sleep 1 &
     printf "\r%s%02d:%02d:%02d" \
     printf "\r%s%02d:%02d:%02d" "${PREFIX}" \
      "${PREFIX}" $((secs/3600)) $(( (secs/60)%60)) $((secs%60))
      $((secs/3600)) $(( (secs/60)%60)) $((secs%60))
     secs=$(( $secs - 1 ))
     secs=$(( $secs - 1 ))
     wait
     wait

Version vom 27. Mai 2011, 21:17 Uhr

Der Shell-Befehl while bewirkt eine Schleife, welche solange wiederholt wird wie eine anfangs stehende Bedingung erfolgreich ist (also Exit-Status 0 liefert). Die Abwicklung der Schleife wird dabei vom Exit-Status eines Befehls gesteuert.

Syntax

Die Shell führt Befehlsliste2 nur dann aus, wenn der letzte Befehl aus Befehlsliste1 den Exit-Status 0 geliefert hat. Nach jedem Schleifendurchlauf führt die Shell Befehlsliste1 erneut aus.

while Befehlsliste1
  do
    Befehlsliste2
  done

Verwendung

Alle 30 Sekunden soll ein "x" auf den Bildschirm geschrieben werden. Das kann nützlich sein, um eine SSH-Verbindung auf einem zweiten Bildschirm am Leben zu erhalten, wenn etwa auf dem Router "dialer idle-timeout 90" eingestellt ist.

$ while true; do echo -n "x"; sleep 30; done

...

$ x=0
$ while test $x -lt 3
> do
> let x+=1
> echo ${x}
> done
1
2
3

$ while true
> do
> banner $(date '+%H:%M')
> sleep 60
> done

Mit der folgenden Schleife können beliebig viele Parameter ausgewertet werden. Durch "( $@ )" wird dabei die Parameterkette an den Leerzeichen in einzelne Abschnitte aufgespalten un nacheinander in ein eindimensionales Array gespeichert.

#!/bin/bash
declare -i i=0
array=( $@ )
while [ $i -le $# ]; do
  echo "Inhalt von array[$i]: ${array[$i]}"
  i=$i+1
done

$ ./while.sh 1 2 "3 4 5"
Inhalt von array[0]: 1
Inhalt von array[1]: 2
Inhalt von array[2]: 3
Inhalt von array[3]: 4

Auf den ersten Eintrag könnte auch mit "echo $array[0]" zugegriffen werden.

Countdown

function countdown
{
  local OLD_IFS="${IFS}"
  IFS=":"
  local ARR=( $1 ) ; shift
  IFS="${OLD_IFS}"
  local PREFIX="$*" ; [ -n "${PREFIX}" ] && PREFIX="${PREFIX} > "
  local SECONDS=$((  (ARR[0] * 60 * 60) + (ARR[1] * 60) + ARR[2]  ))
  local START=$(date +%s)
  local END=$((START + SECONDS))
  local CUR=$START

  while [[ $CUR -lt $END ]]
    do
    CUR=$(date +%s)
    LEFT=$((END-CUR))
    printf "\r%s%02d:%02d:%02d" "${PREFIX}" \
      $((LEFT/3600)) $(( (LEFT/60)%60)) $((LEFT%60))
    sleep 1
    done
  echo "        "
}

countdown "00:00:30" Gleich geht es weiter
function countdown2()
{
  IFS=:
  set -- $*
  secs=$(( ${1#0} * 3600 + ${2#0} * 60 + ${3#0} ))
  PREFIX="$4" ; [ -n "${PREFIX}" ] && PREFIX="${PREFIX}: "
  while [ $secs -gt 0 ]
  do
    sleep 1 &
    printf "\r%s%02d:%02d:%02d" "${PREFIX}" \
      $((secs/3600)) $(( (secs/60)%60)) $((secs%60))
    secs=$(( $secs - 1 ))
    wait
  done
  printf "\r%s%02d:%02d:%02d" "${PREFIX}"
  echo
}
countdown2 "00:00:03" "Auf geht's"