Wiki Ubuntu-it

Indice
Partecipa
FAQ
Wiki Blog
------------------
Ubuntu-it.org
Forum
Chiedi
Chat
Cerca
Planet
  • Pagina non alterabile
  • Informazioni
  • Allegati
  • Differenze per "Programmazione/LinguaggioBash"
Differenze tra le versioni 13 e 38 (in 25 versioni)
Versione 13 del 04/06/2007 22.31.17
Dimensione: 11850
Commento: Aggiunta Programmi utili
Versione 38 del 15/02/2022 08.20.19
Dimensione: 12267
Commento:
Le cancellazioni sono segnalate in questo modo. Le aggiunte sono segnalate in questo modo.
Linea 1: Linea 1:
## page was renamed from LinguaggioBash
Linea 3: Linea 4:
[[BR]]
||<tablestyle="float:right; font-size: 0.9em; width:30%; background:#F1F1ED; margin: 0 0 1em 1em;" style="padding:0.5em;">'''Indice'''[[BR]][[TableOfContents(2)]]||
<<BR>>
<<Indice(depth=1)>>
<<Informazioni(forum="http://forum.ubuntu-it.org/viewtopic.php?f=46&t=620802";)>>
Linea 7: Linea 9:
'''Bash''' (acronimo per '''''b'''ourne '''a'''gain '''s'''hell'') è una '''shell''' del progetto '''GNU''' usata nei sistemi operativi ''Unix'' e specialmente in '''Gnu/Linux''', si tratta di un interprete di comandi che permette all'utente di comunicare col sistema operativo attraverso una serie di funzioni predefinite, o di eseguire programmi.

Tecnicamente '''bash''' è un clone evoluto della ''shell'' standard di Unix (`/bin/sh`) chiamata anche ''Bourne Shell'' dal nome del suo autore originario ''Stephen Bourne'' e per questo il nome '''bash''' deriva dal gioco di parole (definito pessimo addirittura da [http://it.wikipedia.org/wiki/Linus_Torvalds Linus Torvalds]) ''Bourne again shell'' che letteralmente significa ''un'altra shell Bourne'', ma dove ''Bourne again'' può essere interpretato come ''rinata'' creando la definizione finale ''shell rinata''.

In pratica '''bash''' può essere considerata come un processore di macro. Essa è infatti in grado di eseguire i comandi che le vengono passati o di interpretare un semplice linguaggio di programmazione che permette di combinare questi comandi tra di loro per svolgere compiti più complessi. Qualora dovesse presentarsi la necessità la ''shell'' mette a disposizione la possibilità di raccogliere in file delle serie di comandi creando così script eseguibili contenenti variabili, funzioni e controlli di flusso come nei più comuni linguaggi di programmazione.

Una delle altre maggiori funzionalità di '''bash''' è quella della redirezione dell'input e dell'output grazie al quale è possibile eseguire più programmi in cascata passando come input dell'uno l'output dell'altro.

'''GNU Bash''' è la ''shell *nix'' probabilmente più ricca di funzionalità sia per l'uso interattivo che per lo ''scripting''. Spesso però molte funzioni sono poco note.

Questa pagina ha lo scopo di raccogliere il più ampio numero di "trucchi" più o meno noti riguardanti '''Bash''', senza avere peraltro la pretesa di coprire tutte le funzionalità.

= Scripting =
== Variabili ==
Le variabili in ''bash scripting'' non hanno bisogno di inizializzazione, l'assegnazione avviene utilizzando la sintassi:

'''Bash''' (acronimo per '''b'''ourne '''a'''gain '''sh'''ell) è una '''[[https://it.wikipedia.org/wiki/Shell_(informatica)|shell]]''' del progetto '''[[https://it.wikipedia.org/wiki/Progetto_GNU|GNU]]''' usata nei sistemi operativi '''[[https://it.wikipedia.org/wiki/Unix-like|Unix-like]]''' e specialmente in '''GNU/Linux'''.<<BR>>
Si tratta di un interprete di comandi che permette all'utente di comunicare col sistema operativo attraverso una serie di funzioni predefinite e di eseguire programmi.

Tecnicamente '''bash''' è un clone evoluto della shell standard di Unix (`/bin/sh`), chiamata anche ''Bourne Shell'' da Stephen Bourne, nome del suo autore originario. Il nome esteso ''Bourne again shell'' («''un'altra shell Bourne''») contiene un gioco di parole in quanto la sua pronuncia in lingua inglese risulta simile a «''Born again shell''» («''shell rinata''»).

In pratica '''Bash''' può essere considerata come un processore di macro. Alcune delle sue principali caratteristiche sono le seguenti:

 * Può essere eseguita sia tramite dei comandi digitati nel [[AmministrazioneSistema/Terminale|terminale]], sia tramite degli appositi [[Programmazione/Script|script]] (combinando così più comandi tra di loro, per svolgere compiti più complessi).
 * Possibilità, se necessario, di raccogliere in file una serie di comandi (in modo tale da creare script eseguibili contenenti variabili, funzioni e controlli di flusso, come nei più comuni linguaggi di programmazione).
 * Possibilità di ridirezione dell'input e dell'output (''redirect''), che consente di eseguire più programmi "a cascata", dove l'output di un comando può diventare l'input di un comando successivo.

'''Bash''' è qindi uno strumento ricco di funzionalità, che tuttavia potrebbero essere poco note agli utenti novizi. Per questo la guida si propone come una raccolta (non esaustiva) di consigli e "trucchi" relativi all'utilizzo dello scripting in '''Bash'''.

= Variabili =

Le variabili in ''bash scripting'' non hanno bisogno di inizializzazione. L'assegnazione avviene usando la forma:
Linea 25: Linea 29:
è molto importante che il valore sia scritto subito dopo l'uguale senza nessuno spazio.
Il riferimento a tutte le variabili si fanno con anteponendo il carattere ''$'' al nome della variabile:
È molto importante che il valore sia scritto subito dopo l'uguale senza nessuno spazio.
I riferimenti a tutte le variabili si scrivono anteponendo il carattere '''$''' al nome della variabile:
Linea 31: Linea 35:
=== Variabili speciali ===

 *'''$N''' dove N è un numero intero, corrisponde all'argomento N-esimo passato da terminale al programma ($0 è il nome del programma)

 *'''$*''' stringa che contiene tutti gli argomenti passati da terminale al programma

 *'''$#''' contiene il numero di argomenti passati da terminale al programma ($0 escluso)

 * '''$@''' Contiene la lista dei paramentri passati allo script corrente. Ogni parametro viene opportunamente quotato e questo permette l'utilizzo di questa variabile nei cicli for per processare (ad esempio) una lista di nomi di file che possono contenere anche spazi. L'uso di questa variabile è quindi in genere consigliato rispetto a $* che ha la stessa funzione ma non quota i vari parametri.

= Variabili speciali =

 * '''$N''', dove N è un numero intero, corrisponde all'argomento N-esimo passato da terminale al programma ($0 è il nome del programma).

 * '''$*''' stringa che contiene tutti gli argomenti passati da terminale al programma.

 * '''$#''' contiene il numero di argomenti passati da terminale al programma ($0 escluso).

 * '''$@''' Contiene la lista dei parametri passati allo script corrente. Ogni parametro viene opportunamente quotato e questo permette l'utilizzo di questa variabile nei cicli ''for'' per processare, ad esempio, una lista di nomi di file che possono contenere anche spazi. L'uso di questa variabile è quindi in genere consigliato rispetto a $* che ha la stessa funzione ma non quota i vari parametri.
Linea 43: Linea 48:
 *'''$$''' contiene il PID del processo in esecuzione

 *'''$!''' contiene il PID dell'ultimo job in background

== Esecuzione comandi ==
Dato che non è un vero linguaggio di programmazione, molte funzionalità non sono incluse nel ''bash scripting'' ma sono comunque utilizzabili tramite dei programmi esterni.
Per prima cosa è utile sapere che se si vuole "catturare" l'output di un programma ed, ad esempio, inserirlo in una variabile allora si utilizza la sintassi:
 * '''$$''' contiene il PID del processo in esecuzione.

 * '''$!''' contiene il PID dell'ultimo job in background.

= Esecuzione comandi =

Dato che non è un vero linguaggio di programmazione, molte funzionalità non sono incluse nel ''bash scripting'' ma sono comunque usabili tramite dei programmi esterni.

Per prima cosa è utile sapere che per leggere l'output di un programma e, per esempio, inserirlo in una variabile si usa la forma:
Linea 54: Linea 61:
tramite i caratteri di back-quote (ALT GR + ') l'interprete bash esegue il comando e sostituisce con l'output tutta la chiamata; in questo caso, al posto di `uname -r`, è equivalente una scrittura del tipo:
{{{
VAR=2.6.21
oppure:
{{{
VAR=$(uname -r)
echo $(pwd)
}}}
Tramite i caratteri di back-quote ('''ALT GR + ' ''') o il carattere dollaro ('''Shift + 4''') l'interprete bash esegue il comando e sostituisce con l'output tutta la chiamata. In questo caso, al posto di ''{{{`uname -r`}}}'' o ''$(uname -r)'', è equivalente una scrittura del tipo:
{{{
VAR=5.13.0-28
Linea 59: Linea 71:
=== Programmi utili ===

Per avere maggiori informazioni e spiegazione sull'uso di ogni programma vi rimando alle pagine del manuale di ogni comando.

 * '''expr''': dato che non si possono eseguire semplici operazioni matematiche direttamente tramite l'interprete questo comando ci viene in aiuto

 * '''grep''': utile per mostrare solo alcune righe di un determinato output

 * '''cut''': serve per "tagliare" una stringa in tante sotto-stringhe in base ad un determinato separatore, per i programmatori java è molto simile al Tokenizer
 
 * '''cat''': data l'assenza di funzioni proprie dell'interprete per l'apertura e la lettura dei file, questo comando apre il file e stampa il suo contenuto, l'output stampato può essere reindirizzato in una variabile come spiegato sopra

== Manipolazione delle stringhe ==

Nelle ''shell *nix'', storicamente, la manipolazione delle stringhe viene fatto attraverso programmi esterni alla ''shell'' come '''seq''', '''awk''' e '''perl'''. Questi programmi vengono ancora usati quando si vuole mantenere la compatibilità con la '''shell sh''', tuttavia imparare anche il solo '''sed''' (il più semplice dei tre) non è cosa immediata.

Se si usa '''Bash''', non è necessario usare nessun programma esterno, ma basta imparare i tre operatori fondamentali ed alcuni concetti di base, per poter fare tutte le manipolazioni più comuni.

In '''Bash''' una stringa non è altro che una variabile, si indica dunque con '''${nome}''' oppure con la forma abbreviata '''$nome'''. Il nome dentro le graffe può essere seguito da un modificatore che manipola la variabile, ad esempio:

== Programmi utili ==

 * '''expr''': dato che non si possono eseguire semplici operazioni matematiche direttamente tramite l'interprete, tale comando può essere d'aiuto.

 * '''grep''': utile per mostrare solo alcune righe di un determinato output.

 * '''cut''': serve per "tagliare" una stringa in tante sotto-stringhe in base ad un determinato separatore. È molto simile allo !StringTokenizer in java.
 
 * '''cat''': data l'assenza di funzioni proprie dell'interprete per l'apertura e la lettura dei file, questo comando apre il file e stampa il suo contenuto. L'output stampato può essere reindirizzato in una variabile come spiegato sopra.

Per maggiori informazioni sui singoli comandi si rimanda ai manuali dei relativi programmi.

= Manipolazione delle stringhe =

Tradizionalmente, nelle shell '''Unix-like''', la manipolazione delle stringhe viene fatta attraverso programmi esterni alla '''shell''' come '''sed''', '''awk''' e '''perl'''.

Questi programmi vengono ancora usati quando si vuole mantenere la compatibilità con la '''shell sh'''. Tuttavia imparare anche il solo '''sed''' (il più semplice dei tre) non è cosa immediata.

Se si usa '''Bash''' non è necessario usare nessun programma esterno, ma basta imparare i tre operatori fondamentali e alcuni concetti di base per poter eseguire tutte le manipolazioni più comuni.

In '''Bash''' una stringa non è altro che una variabile. Dunque si indica con '''${nome}''' oppure con la forma abbreviata '''$nome'''. Il nome dentro le graffe può essere seguito da un modificatore che manipola la variabile, ad esempio:
Linea 86: Linea 101:
 * '''#''': sottrae dall'inizio della stringa (minimale) 
 * '''%''': sottrae dalla fine della stringa (minimale)
 *'''/ ''': sostituisce una sottostringa con un altra (solo la prima volta che viene incontrata)

Questi operatori sono minimali, questo vuol dire che se si usano le espressioni regolari per indicare la sottostringa (da eliminare o sostituire) verrà individuata in caso di ambiguità la sottostringa più piccola (o solo la prima nel caso della sostituzione).
 * '''#''': sottrae dall'inizio della stringa (minimale).
 * '''%''': sottrae dalla fine della stringa (minimale).
 *'''/ ''': sostituisce una sotto-stringa con un'altra (solo la prima volta che viene incontrata).

Questi operatori sono minimali. Ciò vuol dire che se si usano le espressioni regolari per indicare la sotto-stringa (da eliminare o sostituire) verrà individuata in caso di ambiguità la sotto-stringa più piccola (o solo la prima nel caso della sostituzione).
Linea 93: Linea 108:
 * '''##''': sottrae dall'inizio della stringa (massimale) 
 * '''%%''': sottrae dalla fine della stringa (massimale)
 * '''//''': sostituisce una sottostringa con un altra (tutte le volte che viene incontrata) 

Gli operatori massimali cercano di individuare la sottostringa più grande che corrisponde all'espressione regolare (nel caso del modificatore '''//''' tutte le sottostringhe vengono sostituite). Gli operatori di questo tipo vengono comunemente chiamati anche ''greedy'' (ingordi).

Per una spiegazione dettagliata di tutti i modificatori e anche di altri modi di manipolare le stringhe in '''Bash''' (ad esempio '''expr''') vedere:
[http://www.tldp.org/LDP/abs/html/string-manipulation.html Advanced Bash-Scripting Guide: Manipulating Strings].

===== Esempio: Manipolazione delle stringhe =====
{{{
 VAR="questa.sarebbe.una.stringa.di.esempio"
 * '''##''': sottrae dall'inizio della stringa (massimale).
 * '''%%''': sottrae dalla fine della stringa (massimale).
 * '''//''': sostituisce una sotto-stringa con un altra (tutte le volte che viene incontrata).

Gli operatori massimali cercano di individuare la sotto-stringa più grande che corrisponde all'espressione regolare (nel caso del modificatore '''//''' tutte le sotto-stringhe vengono sostituite). Gli operatori di questo tipo vengono comunemente detti anche ''greedy'' («ingordi»).

Per ulteriori informazioni più dettagliate su tutti i modificatori e le altre modalità di manipolazione delle stringhe in '''Bash''' consultare [[http://www.tldp.org/LDP/abs/html/string-manipulation.html|questa guida]].

== Esempio: Manipolazione delle stringhe ==
{{{
 VAR="questa.è.una.stringa.di.esempio"
Linea 108: Linea 122:
 echo ${VAR#*.}   # --> sarebbe.una.stringa.di.esempio  echo ${VAR#*.} # --> è.una.stringa.di.esempio
Linea 111: Linea 125:
 echo ${VAR%.*}  # --> questa.sarebbe.una.stringa.di
 echo ${VAR%%.*} # --> questa
 
 echo ${VAR/st/ST} # --> queSTa.sarebbe.una.stringa.di.esempio
 echo ${VAR//st/ST} # --> queSTa.sarebbe.una.STringa.di.esempio
}}}

=== Alternativa a basename ===
Quando in uno script ci si deve riferire al nome dello script stesso è usuale utilizzare il comando (esterno a bash) basename. Tuttavia, tramite i modificatori del paragrafo precedente, Bash stessa è in grado di fornire questa funzionalità. Basta usare l'espressione ${0##*/}.

Esempio:
 echo ${VAR%.*} # --> questa.è.una.stringa.di
 echo ${VAR%%.*}  # --> questa
 
 echo ${VAR/st/ST} # --> queSTa.è.una.stringa.di.esempio
 echo ${VAR//st/ST} # --> queSTa.è.una.STringa.di.esempio
}}}

== Alternativa a basename ==

Quando in uno script occorre fare riferimento al nome dello script stesso, è usuale utilizzare il comando '''basename''' (esterno a Bash). Tuttavia, tramite i modificatori del paragrafo precedente, Bash stessa è in grado di fornire questa funzionalità. Basta usare l'espressione ${0##*/}.

Ad esempio:
Linea 129: Linea 144:
===== Esempio: Cambiare l'estensione ai file ===== == Esempio: cambiare l'estensione ai file ==
Linea 133: Linea 148:
== Condizioni ==
Le operazioni condizionali in bash possono essere eseguite in due modi, tramite il blocco ''if..then..else'' (simile al suo omonimo in c) o tramite il blocco ''case..in'' (simile allo switch).
=== If..Then..Else ===
Tutto il lavoro di questo blocco è svolto dal programma bash '''test''', il suo uso è:

= Condizioni =
Le operazioni condizionali in '''bash''' possono essere eseguite in due modi: tramite il blocco '''if..then..else''' (simile al suo omonimo in linguaggio c), oppure tramite il blocco '''case..in''' (simile allo switch).

== If..Then..Else ==

Tutto il lavoro di questo blocco è svolto dal programma bash '''test'''. Il suo uso è:
Linea 141: Linea 159:
ci sono molte espressioni possibili le più comuni sono:
 * '''-f nomefile''': ritorna vero se il file esiste
 * '''-d directory''': ritorna vero se la directory esiste
 * '''str1 = str2''': ritorna vero se la str1 è uguale alla str2
 * '''str1 != str2''': ritorna vero se la str1 è diversa dalla str2
 * '''arg1 -eq arg2''': confronto numerico, ritorna vero se arg1 è uguale a arg2
 * '''arg1 -ne arg2''': confronto numerico, ritorna vero se arg1 è diverso a arg2
 * '''arg1 -lt arg2''': confronto numerico, ritorna vero se arg1 è minore a arg2 ('''-le''' per maggiore o uguale)
 * '''arg1 -gt arg2''': confronto numerico, ritorna vero se arg1 è maggiore a arg2 ('''-ge''' per maggiore o uguale)
 * '''espr1 -o espr2''': concatena più condizioni, uguale all'or, ritorna vero o se l'espr1 o se l'espr2 ritorna vero
 * '''espr1 -a espr2''': concatena più condizioni, uguale all'and, ritorna vero se sia l'espr1 che l'espr2 ritornano vero

Questo è un esempio d'uso del ciclo if:
ci sono molte espressioni possibili, di cui le più comuni sono:
 * '''-f nomefile''': ritorna vero se il file esiste.
 * '''-d directory''': ritorna vero se la directory esiste.
 * '''str1 = str2''': ritorna vero se la str1 è uguale alla str2.
 * '''str1 != str2''': ritorna vero se la str1 è diversa dalla str2.
 * '''arg1 -eq arg2''': confronto numerico, ritorna vero se arg1 è uguale a arg2.
 * '''arg1 -ne arg2''': confronto numerico, ritorna vero se arg1 è diverso a arg2.
 * '''arg1 -lt arg2''': confronto numerico, ritorna vero se arg1 è minore a arg2 ('''-le''' per maggiore o uguale).
 * '''arg1 -gt arg2''': confronto numerico, ritorna vero se arg1 è maggiore a arg2 ('''-ge''' per maggiore o uguale).
 * '''espr1 -o espr2''': concatena più condizioni, uguale all'or, ritorna vero o se l'espr1 o se l'espr2 ritorna vero.
 * '''espr1 -a espr2''': concatena più condizioni, uguale all'and, ritorna vero se sia l'espr1 che l'espr2 ritornano vero.

Questo è un esempio d'uso del costrutto if:
Linea 163: Linea 181:
Come potete notare inserire tra parentesi quadre l'espressione da testare è equivalente a scrivere:
Un'altra forma del costrutto ''if'':

{{{
if [ -f prova.txt ];then echo Il file esiste!;else echo Il file non esiste!;fi
}}}

Come si può notare, inserire tra parentesi quadre l'espressione da testare è equivalente a scrivere:
Linea 167: Linea 192:
Attenzione: l'if bash è molto rigoroso dal punto di vista sintattico, dopo l'espressione da valutare dovete andare a capo e scrivere la parola ''then'', notereta anche che non c'è bisogno di parentesi graffe, infatti il blocco viene terminato con la parola chiave ''fi''.

=== Case..In ===
L'if bash è molto rigoroso dal punto di vista sintattico. Dopo l'espressione da valutare è necessario andare o a capo e scrivere la parola chiave ''then'', oppure frapporre un ''';''' e scrivere di seguito la parola chiave.

Si noti inoltr
e che non c'è bisogno di parentesi graffe, infatti il blocco viene terminato con la parola chiave ''fi''.

== Case..in ==
Linea 180: Linea 208:
viene valutata $a ed in base al valore contenuto vengono eseguite le operazioni associate.
== Cicli Enumerativi ==
Al contrario di altri linguaggi, proprio per l'utilizzo che si fà del Bash Scripting, i cicli sono del tipo enumerativo, cioè eseguono le operazioni per tutti gli elementi presenti un una lista indicata, il più importante è sicuramente il ciclo ''for..do''.
=== For..Do ===
La sua sintassi è:
viene valutata '''$a''' e in base al valore contenuto vengono eseguite le operazioni associate.<<BR>>
Si noti inoltre che non c'è bisogno di parentesi graffe, infatti il blocco viene terminato con la parola chiave ''esac''.

= Cicli enumerativi =

Al contrario di altri linguaggi, nel Bash Scripting i cicli sono del tipo enumerativo, cioè eseguono le operazioni per tutti gli elementi presenti un una lista indicata. Il più importante è sicuramente il ciclo ''for..do''.

== For..do ==

La sua forma è:
Linea 191: Linea 224:
Il parametro dopo ''in'' è proprio la lista di cui parlavo prima, se ad esempio abbiamo un ciclo del tipo: il cui parametro dopo ''in'' rappresenta la lista prima citata.<<BR>>
Nel
seguente esempio il ciclo stamperà sullo standard output ''1 2 3 4 5'':
Linea 198: Linea 232:
il ciclo stamperà sullo standard output ''1 2 3 4 5'', altro esempio:
{{{
for i in `ls`

In questo altro esempio
il ciclo stamperà una lista di file:
{{{
for i in *
Linea 205: Linea 240:
questo ciclo stamperà una lista di file, cioè il ciclo eseguirà per ogni singola parola dell'output di '''ls''' l'operazione di stampa a video.
Linea 207: Linea 242:
 * [http://www.gnu.org/software/bash/manual/bash.html Manuale ufficiale di Bash]
 * [http://www.tldp.org/LDP/abs/html/ La ''bibbia'' del ''bash scripting'']

 * [[https://www.gnu.org/software/bash/|Sito ufficiale del progetto]]
 * [[https://it.wikipedia.org/wiki/Bash|Pagina enciclopedica]]
 * [[http://www.gnu.org/software/bash/manual/bash.html|Manuale ufficiale di Bash]]
 * [[http://www.tldp.org/LDP/abs/html/|Guida avanzata di scripting Bash]]
 * [[http://www.pluto.it/files/ildp/guide/abs/index.html|Guida avanzata di scripting Bash in italiano]]


Problemi in questa pagina? Segnalali in questa discussione

Introduzione

Bash (acronimo per bourne again shell) è una shell del progetto GNU usata nei sistemi operativi Unix-like e specialmente in GNU/Linux.
Si tratta di un interprete di comandi che permette all'utente di comunicare col sistema operativo attraverso una serie di funzioni predefinite e di eseguire programmi.

Tecnicamente bash è un clone evoluto della shell standard di Unix (/bin/sh), chiamata anche Bourne Shell da Stephen Bourne, nome del suo autore originario. Il nome esteso Bourne again shellun'altra shell Bourne») contiene un gioco di parole in quanto la sua pronuncia in lingua inglese risulta simile a «Born again shell» («shell rinata»).

In pratica Bash può essere considerata come un processore di macro. Alcune delle sue principali caratteristiche sono le seguenti:

  • Può essere eseguita sia tramite dei comandi digitati nel terminale, sia tramite degli appositi script (combinando così più comandi tra di loro, per svolgere compiti più complessi).

  • Possibilità, se necessario, di raccogliere in file una serie di comandi (in modo tale da creare script eseguibili contenenti variabili, funzioni e controlli di flusso, come nei più comuni linguaggi di programmazione).
  • Possibilità di ridirezione dell'input e dell'output (redirect), che consente di eseguire più programmi "a cascata", dove l'output di un comando può diventare l'input di un comando successivo.

Bash è qindi uno strumento ricco di funzionalità, che tuttavia potrebbero essere poco note agli utenti novizi. Per questo la guida si propone come una raccolta (non esaustiva) di consigli e "trucchi" relativi all'utilizzo dello scripting in Bash.

Variabili

Le variabili in bash scripting non hanno bisogno di inizializzazione. L'assegnazione avviene usando la forma:

VAR=valore

È molto importante che il valore sia scritto subito dopo l'uguale senza nessuno spazio. I riferimenti a tutte le variabili si scrivono anteponendo il carattere $ al nome della variabile:

VAR=world
echo hello $VAR!

Variabili speciali

  • $N, dove N è un numero intero, corrisponde all'argomento N-esimo passato da terminale al programma ($0 è il nome del programma).

  • $* stringa che contiene tutti gli argomenti passati da terminale al programma.

  • $# contiene il numero di argomenti passati da terminale al programma ($0 escluso).

  • $@ Contiene la lista dei parametri passati allo script corrente. Ogni parametro viene opportunamente quotato e questo permette l'utilizzo di questa variabile nei cicli for per processare, ad esempio, una lista di nomi di file che possono contenere anche spazi. L'uso di questa variabile è quindi in genere consigliato rispetto a $* che ha la stessa funzione ma non quota i vari parametri.

  • $? Contiene il valore di uscita dell'ultimo comando o funzione. Il comando ha successo se ritorna zero, qualsiasi altro valore indica invece un codice di errore.

  • $$ contiene il PID del processo in esecuzione.

  • $! contiene il PID dell'ultimo job in background.

Esecuzione comandi

Dato che non è un vero linguaggio di programmazione, molte funzionalità non sono incluse nel bash scripting ma sono comunque usabili tramite dei programmi esterni.

Per prima cosa è utile sapere che per leggere l'output di un programma e, per esempio, inserirlo in una variabile si usa la forma:

VAR=`uname -r`
echo `pwd`

oppure:

VAR=$(uname -r)
echo $(pwd)

Tramite i caratteri di back-quote (ALT GR + ' ) o il carattere dollaro (Shift + 4) l'interprete bash esegue il comando e sostituisce con l'output tutta la chiamata. In questo caso, al posto di `uname -r` o $(uname -r), è equivalente una scrittura del tipo:

VAR=5.13.0-28
echo /usr/src/linux

Programmi utili

  • expr: dato che non si possono eseguire semplici operazioni matematiche direttamente tramite l'interprete, tale comando può essere d'aiuto.

  • grep: utile per mostrare solo alcune righe di un determinato output.

  • cut: serve per "tagliare" una stringa in tante sotto-stringhe in base ad un determinato separatore. È molto simile allo StringTokenizer in java.

  • cat: data l'assenza di funzioni proprie dell'interprete per l'apertura e la lettura dei file, questo comando apre il file e stampa il suo contenuto. L'output stampato può essere reindirizzato in una variabile come spiegato sopra.

Per maggiori informazioni sui singoli comandi si rimanda ai manuali dei relativi programmi.

Manipolazione delle stringhe

Tradizionalmente, nelle shell Unix-like, la manipolazione delle stringhe viene fatta attraverso programmi esterni alla shell come sed, awk e perl.

Questi programmi vengono ancora usati quando si vuole mantenere la compatibilità con la shell sh. Tuttavia imparare anche il solo sed (il più semplice dei tre) non è cosa immediata.

Se si usa Bash non è necessario usare nessun programma esterno, ma basta imparare i tre operatori fondamentali e alcuni concetti di base per poter eseguire tutte le manipolazioni più comuni.

In Bash una stringa non è altro che una variabile. Dunque si indica con ${nome} oppure con la forma abbreviata $nome. Il nome dentro le graffe può essere seguito da un modificatore che manipola la variabile, ad esempio:

 VAR="stringa-di-esempio"
 echo ${VAR#stringa-}

elimina "stringa-" dall'inizio della variabile.

I modificatori sono molti, ma possono essere facilmente ricordati se si imparano i tre fondamentali:

  • #: sottrae dall'inizio della stringa (minimale).

  • %: sottrae dalla fine della stringa (minimale).

  • / : sostituisce una sotto-stringa con un'altra (solo la prima volta che viene incontrata).

Questi operatori sono minimali. Ciò vuol dire che se si usano le espressioni regolari per indicare la sotto-stringa (da eliminare o sostituire) verrà individuata in caso di ambiguità la sotto-stringa più piccola (o solo la prima nel caso della sostituzione).

Per ottenere gli operatori massimali basta raddoppiare il simbolo:

  • ##: sottrae dall'inizio della stringa (massimale).

  • %%: sottrae dalla fine della stringa (massimale).

  • //: sostituisce una sotto-stringa con un altra (tutte le volte che viene incontrata).

Gli operatori massimali cercano di individuare la sotto-stringa più grande che corrisponde all'espressione regolare (nel caso del modificatore // tutte le sotto-stringhe vengono sostituite). Gli operatori di questo tipo vengono comunemente detti anche greedy («ingordi»).

Per ulteriori informazioni più dettagliate su tutti i modificatori e le altre modalità di manipolazione delle stringhe in Bash consultare questa guida.

Esempio: Manipolazione delle stringhe

 VAR="questa.è.una.stringa.di.esempio"
 
                                    # Risultato:
 
 echo ${VAR#*.}      # --> è.una.stringa.di.esempio
 echo ${VAR##*.}     # --> esempio
 
 echo ${VAR%.*}      # --> questa.è.una.stringa.di
 echo ${VAR%%.*}     # --> questa
 
 echo ${VAR/st/ST}   # --> queSTa.è.una.stringa.di.esempio
 echo ${VAR//st/ST}  # --> queSTa.è.una.STringa.di.esempio

Alternativa a basename

Quando in uno script occorre fare riferimento al nome dello script stesso, è usuale utilizzare il comando basename (esterno a Bash). Tuttavia, tramite i modificatori del paragrafo precedente, Bash stessa è in grado di fornire questa funzionalità. Basta usare l'espressione ${0##*/}.

Ad esempio:

usage () {
        echo "usage: ${0##*/} "
        exit 1
}

Esempio: cambiare l'estensione ai file

Rinomina tutti i file *.txt della directory corrente in *.log:

for f in *.txt; do mv "$f" "${f/%txt/log}"; done

Condizioni

Le operazioni condizionali in bash possono essere eseguite in due modi: tramite il blocco if..then..else (simile al suo omonimo in linguaggio c), oppure tramite il blocco case..in (simile allo switch).

If..Then..Else

Tutto il lavoro di questo blocco è svolto dal programma bash test. Il suo uso è:

test [espr]

ci sono molte espressioni possibili, di cui le più comuni sono:

  • -f nomefile: ritorna vero se il file esiste.

  • -d directory: ritorna vero se la directory esiste.

  • str1 = str2: ritorna vero se la str1 è uguale alla str2.

  • str1 != str2: ritorna vero se la str1 è diversa dalla str2.

  • arg1 -eq arg2: confronto numerico, ritorna vero se arg1 è uguale a arg2.

  • arg1 -ne arg2: confronto numerico, ritorna vero se arg1 è diverso a arg2.

  • arg1 -lt arg2: confronto numerico, ritorna vero se arg1 è minore a arg2 (-le per maggiore o uguale).

  • arg1 -gt arg2: confronto numerico, ritorna vero se arg1 è maggiore a arg2 (-ge per maggiore o uguale).

  • espr1 -o espr2: concatena più condizioni, uguale all'or, ritorna vero o se l'espr1 o se l'espr2 ritorna vero.

  • espr1 -a espr2: concatena più condizioni, uguale all'and, ritorna vero se sia l'espr1 che l'espr2 ritornano vero.

Questo è un esempio d'uso del costrutto if:

if [ -f prova.txt ]
 then
       echo Il file esiste!
  else
       echo Il file non esiste!
fi

Un'altra forma del costrutto if:

if [ -f prova.txt ];then echo Il file esiste!;else echo Il file non esiste!;fi

Come si può notare, inserire tra parentesi quadre l'espressione da testare è equivalente a scrivere:

test -f prova.txt

L'if bash è molto rigoroso dal punto di vista sintattico. Dopo l'espressione da valutare è necessario andare o a capo e scrivere la parola chiave then, oppure frapporre un ; e scrivere di seguito la parola chiave.

Si noti inoltre che non c'è bisogno di parentesi graffe, infatti il blocco viene terminato con la parola chiave fi.

Case..in

L'uso del blocco case è molto semplice:

case $a in 
 1) echo sono un 1;;
 2) echo sono un 2;;
 3) echo sono un 3;;
 *) echo Non sono ne un 1 ne un 2 ne un 3;;
esac

viene valutata $a e in base al valore contenuto vengono eseguite le operazioni associate.
Si noti inoltre che non c'è bisogno di parentesi graffe, infatti il blocco viene terminato con la parola chiave esac.

Cicli enumerativi

Al contrario di altri linguaggi, nel Bash Scripting i cicli sono del tipo enumerativo, cioè eseguono le operazioni per tutti gli elementi presenti un una lista indicata. Il più importante è sicuramente il ciclo for..do.

For..do

La sua forma è:

for [VAR] [in [LIST]]
 do
    [COMANDI]
done

il cui parametro dopo in rappresenta la lista prima citata.
Nel seguente esempio il ciclo stamperà sullo standard output 1 2 3 4 5:

for i in 1 2 3 4 5
 do
    echo sono $i
done

In questo altro esempio il ciclo stamperà una lista di file:

for i in *
 do
    echo Sono il file $i
done

Ulteriori risorse


CategoryProgrammazione