Documentos de Académico
Documentos de Profesional
Documentos de Cultura
Este es uno de los capítulos del tutorial Scripts en Bash. Encontrarás los enlaces a todos los de capítulos, al final de este
artículo.
Como te comenté en el capítulo anterior sobre variables de este tutorial sobre scripting en Bash, hasta el
momento solo has podido hacer scripts lineales. Es decir, tu script empezará en un punto y terminará,
pero no puedes hacer nada por variar el flujo de tu script en función de lo que allí suceda. Sin embargo,
en este nuevo capítulo del tutorial puedes cambiar esta situación. Así, si al realizar una operación Dona para ayudarme con este
obtienes un resultado u otro, podrás tomar una decisión para emprender una u otra acción. Así en este proyecto
capítulo verás los condicionales en Bash.
Con los condicionales en Bash, ya podemos, realmente, empezar a profundizar en el maravilloso mundo
del scripting, y un poquito en la automatización. Por ejemplo, puedes crear un proceso que una vez al
mes te borre los archivos de un directorio si el espacio es superior a uno determinado y si la fecha del
archivo es inferior a una dada. Todo está condicionado…
CONDICIONALES EN BASH
Si tienes alguna idea de programación seguro que has oído hablar de los condicionales. Si no, es posible
que conozcas el servicio ifttt. Eso de que si pasa esto entonces haz aquello. Pues ese servicio no es mas
que una extrapolación de esto de los condicionales. Se trata de realizar una u otra acción en función de
una pregunta. Si el resultado de la pregunta es uno, emprenderás una acción y si el resultado es otro,
emprenderás una acción diferente.
En Bash, esto de los condicionales se materializa con if then else y con case . Pero
independientemente de uno u otro, en cualquier caso, se parte de resolver una cuestión, una prueba, un
test, al final una comparación. De esta manera esto es lo primero que tienes que abordar en este
capítulo, comparaciones en Bash.
COMPARANDO EN BASH
Para comparar en Bash se utiliza test o [ . Ambos son totalmente equivalentes, y ambos están
implementados, en el propio Bash. En el caso [ , por cuestiones prácticas es obligatorio también
utilizar su pareja [ o de otra forma te da error. Esta es la razón para que tenga que estar separado el
corchete de la expresión que viene a continuación, porque es un comando.
Por otro lado, también puedes utilizar dobles corchetes [[ , para realizar tus comparaciones. Los
dobles corchetes resultan ser una mejora respecto a los simples. Así, las diferencias entre uno y otro son
las siguientes,
1. No tienes que utilizar las comillas con las variables, los dobles corchetes trabajan perfectamente con
los espacios. Así [ -f "$file" ] es equivalente a [[ -f $file ]] .
2. Con [[ puedes utilizar los operadores || y && , así como < y >` para las
comparaciones de cadena.
4. También puedes utilizar comodines como por ejemplo en la expresión [[ abc = a\* ]]
Es posible que te preguntes por la razón para seguir utilizando [ simple corchete en lugar de doble.
La cuestión es por compatibilidad. Si utilizas Bash en diferentes equipos es posible que te encuentres
alguna imcompatibilidad. Así que depende3 de ti y de donde lo vayas a utilizar. A modo de resumen, aquí
tienes diferentes opciones,
COMPARANDO CADENAS
[[ [
> >
< <
= =
!= !=
COMPARANDO ENTEROS
[[ [
Aquí quiero que te des cuenta que [[ 001 = 1 ]] es falso mientras que [[ 001 -eq 1 ]] es
cierto.
OPERADORES BOOLEANOS
[[ [
&& -a
|| -o
AGRUPACIÓN
Para agrupar operaciones booleanas puedes utilizar paréntesis con los dobles corchetes, mientras que
en el caso de los simples corchetes deberás utilizar los paréntesis pero escapados.
-s con esta opción puedes saber si el tamaño del archivo es mayor que cero. Es decir, que no se
trata de un archivo vacío
Hemos llegado al punto de que hagas tu primera comparación. Para esto tienes que utilizar
if then else . Vamos, si esto se cumple haz aquello y si no, pues haz otra cosa. Dicho así suena
fácil, y realmente lo es. De esta forma. Vamos a establecer una variable a y le vas a asignar un
nombre. Si este nombre termina en a supondremos que es un nombre de mujer, y si termina con
cualquier otra letra supondremos que es de hombre. Como viste en el capítulo anterior del podcast, para
utilizar argumentos, son los $1 a $9 . De esta manera nuestro script tendrá un aspecto como lo
que te indico a continuación,
#!/bin/bash
if [[ $1 =~ (.*)a$ ]]
then
echo Sra. $1
else
echo Sr. $1
fi
De esta forma aprovecho las características que ofrece los dobles corchetes con las expresiones
regulares. No es por complicar la cosa. Otro ejemplo, es para saber si un número es par o impar. Para
ello, puedes hacer un script como el que te muestro a continuación,
#!/bin/bash
if [[ $(($1 % 2)) == 0 ]]
then
echo Par
else
echo Impar
fi
Si te fijas he añadido una sangría, de forma que tanto echo Par como echo Impar están
desplazados hacia la derecha. Esto es mas por cuestiones de legibilidad que por otra razón. No es
necesario hacer esto en Bash. Perfectamente lo podrías haber puesto como,
#!/bin/bash
if [[ $(($1 % 2)) == 0 ]]
then
echo Par
else
echo Impar
fi
Sin embargo, coincidirás conmigo, que esta segunda forma, es bastante mas difícil de leer que en el
caso del primer ejemplo.
Indicarte que puedes utilizar diferentes opciones, dependiendo del objetivo que persigas. Así, puedes
hacer,
if [[ $a > $b ]]
then
echo mayor
fi
if [[ $a > $b ]]
then
echo mayor
else
echo menor
fi
si se cumple algo entonces haz esto en caso contrario si se cumple otra cosa haz lo otro
if [[ $a > $b ]]
then
echo mayor
elif [[ $b > $c ]]
then
fi
Y por supuesto una que combina las anteriores porque contiene todas las posibilidades, y que me niego
a transcribir, porque seguro que me equivoco.
if [[ $a > $b ]]
then
echo "mayor"
elif [[ $b > $c ]]
then
else
fi
IF-THEN ANIDADOS
Por supuesto, es posible el uso de if-then anidados. Un if-then anidado es como una especie de yincana
(no tenía ni idea que se escribiera así). Me refiero a ese tipo de competiciones que están compuestas de
varias pruebas, de forma que cada vez que superas una, tienes una siguiente prueba que superar. Pues
esto es lo mismo. Un if-then anidado, no es mas que una sucesión de comparaciones, de forma que cada
vez que superas uno te tienes, que enfrentar a un siguiente. Por ejemplo
if [[ $a -gt $b ]];then
if [[ $d -gt $e ]];then
fi
fi
fi
fi
fi
CONCATENANDO COMPARACIONES
En ocasiones necesitarás que se cumplan varias condiciones o una combinación de ellas para poder
tomar una u otra acción. Para conseguir esta concatenación de operaciones, tienes que utilizar
operadores booleanos,
y, representado por && . Esto implica que las dos comparaciones tienen que ser ciertas.
o, representado por || . Con que una de las dos comparaciones sea cierta es suficiente.
Por ejemplo, si quieres comprobar que un fichero es de lectura y escritura, tienes que utilizar
[[ -r $f ]] && [[ -w $f ]] . Ahora bien, si te vale con que sea de lectura o escritura, tienes
que tuilizar el operador || quedando en esta ocasión [[ -r $f ]] || [[ -w $f ]]
CONDICIONALES: CASE
Un caso especial es cuando quieres comparar un elemento con diferentes supuestos. Podrías hacer
if-then concatenados como has visto anteriormente, sin embargo, lo que no me puedes negar es
que las concatenaciones son difíciles de leer. Al final se convierten el algo realmente farragoso, no solo
de leer, si no también de trabajar. En algunos casos puedes reemplazar estas concatenaciones por otro
tipo de comparación como son los Case . Lo que sin duda, Case traerá a tu código mucha mas
claridad, según el caso, y resultará mas cómodo leer el código posteriormente.
case <expresión> in
<patrón 1>)
comandos
;;
<patrón 2>)
comandos
;;
*)
comandos
;;
esac
Puedes utilizarla para operaciones realmente sencillas, como podría ser el ejemplo, sobre el significado
de los colores que lees a continuación,
#!/bin/bash
case $1 in
amarillo)
;;
naranja)
;;
rojo)
;;
violeta)
;;
azul)
;;
verde)
;;
blanco)
;;
*)
;;
esac
No solo lo puedes utilizar en este caso, también puedes complicarlo, añadiendo rangos, un ejemplo,
#!/bin/bash
case $1 in
[0-9])
echo "Unidades"
;;
1[0-9]|2[0-9]|3[0-9]|4[0-9]|5[0-9]|6[0-9]|7[0-9]|8[0-
9]|9[0-9])
echo "Decenas"
;;
1[0-9][0-9]|2[0-9][0-9]|3[0-9][0-9]|4[0-9][0-9]|5[0-9][0-
9]|6[0-9][0-9]|7[0-9][0-9]|8[0-9][0-9]|9[0-9][0-9])
echo "Centenas"
;;
1[0-9][0-9][0-9]|2[0-9][0-9][0-9]|3[0-9][0-9][0-9]|4[0-9]
[0-9][0-9]|5[0-9][0-9][0-9]|6[0-9][0-9][0-9]|7[0-9][0-9][0-
9]|8[0-9][0-9][0-9]|9[0-9][0-9][0-9])
echo "Millares"
;;
*)
;;
esac
Probablemente te preguntarás que forma mas extraña de poner rangos. La cuestión es que no estamos
poniendo rangos numéricos, sino que son rango de caracteres, es decir, caracteres que van del 0 al
9 . En este caso, probablemente sea mejor utilizar if-then ,
#!/bin/bash
if [[ $1 -lt 10 ]]
then
echo "Unidades"
then
echo "Decenas"
then
echo "Centenas"
then
echo "Millares"
else
fi
Pero puedes pervertir el uso de Case como tu quieras. Así puedes simplificar el ejemplo anterior,
utilizando otra opción con Case ,
#!/bin/bash
case 1 in
echo "Unidades"
;;
echo "Decenas"
;;
echo "Centenas"
;;
echo "Millares"
;;
*)
;;
esac
Como puedes ver, así que resulta mucho mas sencillo de leer.
Para terminar, otro ejemplo, donde puedes establecer la periodificación de la historia de forma
relativamente sencilla utilizando un Case ,
#!/bin/bash
case 1 in
periodo="Paleolítico inferior"
;;
periodo="Paleolítico medio"
;;
periodo="Paleolítico superior"
;;
periodo="Mesolítico"
;;
periodo="Neolítico"
;;
;;
;;
;;
*)
periodo="Historia"
;;
esac
CONCLUSIONES
Con este segundo capítulo del tutorial sobre scripts en Bash, ya conoces las variables y los
condicionales, tanto if then ese como su primo Case. Lo siguiente es conocer el funcionamiento de los
bucles, para evitar tener que realizar operaciones repetitivas.
Más información,
tldp.org
Ryan Tutorials
Listado de capítulos
Variables en Bash
Bucles en Bash
VARIABLES EN BASH
En este primer capítulo del tutorial sobre scripts aprenderás como trabajar con variables en Bash,
incluso texto y operaciones matemáticas desde un script
BUCLES EN BASH
Los bucles en Bash, no es mas que una forma simple de repetir instrucciones, tantas veces como
necesites, sin tener que escribirlas una y otra vez.
Responder
Hola Eliot,
Un saludo y gracias.
Responder
Tu dirección de correo electrónico no será publicada. Los campos obligatorios están marcados con *
Comentario *
Nombre *
Correo electrónico *
Web
Publicar el comentario