você está aqui: Home  → Colunistas  →  Cantinho do Shell

 

Bash - Estruturas Básicas

Por Rodrigo Bernardo Pimentel

Data de Publicação: 21 de Junho de 2007

É comum queremos executar a mesma função, ou uma função parecida, sobre uma série de argumentos, um por vez. Em bash, há duas estruturas básicas pra isso: for e while.

O for, em bash, é diferente do for em linguagens como C ou Perl. Nessas linguagens, o for é simplesmente um while mais completo. Em bash, o for atua sobre uma seqüência de parâmetros (não necessariamente numéricos ou seqüenciais). Por exemplo:

[rbp@muppets ~]$ for i in 1 2 3 4 5; do echo $i; done
1
2
3
4
5
[rbp@muppets ~]$

Ou,

[root@muppets ~]$ for login in rbp sizinha queiroz; do adduser $login; done
[root@muppets ~]$

Você pode inclusive usar o for a partir de uma saída de comando. Por exemplo, digamos que você tenha um arquivo com uma série de nomes de usuários a serem acrescentados ao sistema, como no exemplo acima:

[root@muppets ~]$ for login in `cat /tmp/usuarios`; do adduser $login; done
[root@muppets ~]$

O while, por sua vez, tem funcionamento semelhante à maioria das linguagens procedurais mais comuns. Um comando ou bloco de comandos continua a ser executado enquanto uma determinada condição for verdadeira. Por exemplo, imitando o exemplo acima:

[rbp@muppets ~]$ while [ $i -le 5 ]; do echo $i; i=$(($i + 1)); done
1
2
3
4
5
[rbp@muppets ~]$

Aos poucos:

while [ $i -le 5 ]

O while deve ser seguido de "verdadeiro" ou "falso". Se for "verdadeiro", o bloco é executado e o teste é repetido. Se for falso, não.

No caso, [ $i -le 5 ] é uma forma de usarmos o comando test. Esse comando testa uma expressão e retorna verdadeiro ou falso. Poderia ser escrito como

while test $i -le 5

$i -le 5 é a expressão testada. O programa test aceita alguns operadores. Entre eles:

-lt (less than) primeiro argumento é menor do que o segundo
-le (less or equal) primeiro argumento é menor ou igual ao segundo
-gt (greater than) primeiro argumento é maior do que o segundo
-ge (greater or equal) primeiro argumento é maior ou igual ao segundo
= primeiro argumento é igual ao segundo
!= primeiro argumento é diferente do segundo

O programa test pode também fazer testes unários, ou seja, com só um argumento. Por exemplo,

arq="/tmp/arquivo"
tmp=$arq
i=1
while [ -e $tmp ]; do
        i=$(($i+1))
        tmp=$arq$i
done
touch $tmp

Esse scriptzinho (note que não o fiz na linha de comando, mas indentado, para usá-lo a partir de um arquivo; funciona dos dois jeitos) só sai do loop quando achar um nome de arquivo que não exista. Ou, de forma mais didática, "enquanto existir (-e) o arquivo cujo nome está na variável $tmp, ele continua executado o bloco de comandos".

Alguns dos operadores unários mais comuns são:

-e arquivo ou diretório existe
-f é um arquivo (em oposição a ser um diretório)
-d é um diretório
-x arquivo tem permissão de execução para o usuário atual
-w arquivo tem permissão de escrita pelo usuário atual
-r arquivo tem permissão de leitura pelo usuário atual

Para mais detalhes, man test.

Continuando:

do echo $i

Logo após um while <teste>, é preciso iniciar o primeiro comando com do. Os seguintes (se houver), não.

A próxima linha mostra um exemplo de algo que não tem nada a ver com a estrutura do while em si, mas é um truquezinho legal de bash:

i=$(($i + 1))

A construção $((expressão)) é um operador matemático em bash. Isso é expandido para o resultado da expressão. Por exemplo,

[rbp@muppets ~]$ echo $((2 + 3)) =
5
[rbp@muppets ~]$ echo $((2 - 3))  # Funciona com números negativos =
-1
[rbp@muppets ~]$ echo $((2 * 3)) =
6
[rbp@muppets ~]$ echo $((10 / 2)) =
5
[rbp@muppets ~]$ echo $((3 / 2))  # Não usa casas decimais =
1
[rbp@muppets ~]$  =

Mas, como diz um amigo meu, voltando...

done

Isso termina "oficialmente" o loop while. A propósito, como pode ter sido notado, termina o for também.

Como foi dito acima, o while espera "verdadeiro" ou "falso". Eu nunca disse que esperava isso só do programa test :)

Com efeito, qualquer expressão pode ser usada, e seu valor de retorno será utilizado para determinar se é "verdadeiro" ou "falso". Para quem não sabe, todo programa em Unix retorna um valor ao terminar sua execução. Normalmente, se tudo correu bem o valor retornado é 0 (zero). Se há algum erro, o valor retornado, diferente de zero, indica o tipo de erro (veja as manpages dos programas; man fetchmail seção exit codes é um bom exemplo). Portanto, ao contrário do que programadores de C ou Perl poderiam achar intuitivo (dentro de um while, ou uma condição em geral), um programa que retorna 0 é considerado "verdadeiro" aos olhos do "while``.

Assim, podemos fazer:

while w | grep -qs rbp; do
        sleep 5s
done
echo 'rbp acaba de sair do sistema!'

Nesse exemplo, o while checa o retorno da expressão w | grep -qs rbp. Isso retorna "verdadeiro" quando o grep acha rbp na saída do comando w. Toda vez que achar, espera 5 segundos e checa de novo. Quando não achar, sai do loop e mostra um aviso de que a última sessão do rbp foi fechada.

Pra finalizar: se você quiser fazer um loop infinito, pode usar : (dois pontos) como condição sempre verdadeira:

while : ; do 
        echo 'Emacs rules!'
done

Isso vai imprimir uma constatação sábia infinitamente, até você usar C-c (Ctrl + C). Normalmente, isso é utilizado com alguma condição de parada.

Esta dica foi publicada originalmente na Dicas-L em 16 de agosto de 2000
Error: No site found with the domain 'moodle.idph.com.br' (Learn more)