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 obtienes un resultado u otro, podrás tomar una decisión para emprender una u otra acción. Así en este 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,
- 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 ]]
. - Con
[[
puedes utilizar los operadores||
y&&
, así como para las comparaciones de cadena. - Puedes utilizar el operador
=~
para expresiones regulares, compo por ejemplo[[ $respuesta =~ ^s(i)?$ ]]
- 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
[[ | [ | |
---|---|---|
-gt | -gt | mayor |
-lt | -lt | menor |
-ge | -ge | mayor o igual |
-le | -le | menor o igual |
-eq | -eq | igual |
-ne | -ne | distinto |
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.
Otros comparadores interesantes
-d
te permitirá saber si es un directorio y si existe-
f lo mismo que en el caso anterior pero para archivos-r
en este caso te permite saber si el archivo tiene permiso de lectura-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-w
te permitirá identificar si el archivo tienen permisos de escritura-x
lo mismo que en el caso anterior pero para el caso de permisos de ejecución.
Condicionales: if then else
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.
Algunas opciones con if-then-else
Indicarte que puedes utilizar diferentes opciones, dependiendo del objetivo que persigas. Así, puedes hacer,
si se cumple algo entonces haz esto
if [[ $a > $b ]]
then
echo mayor
fi
si se cumple algo entonces haz esto en caso contrario haz lo otro
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
echo lo que sea
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
echo "lo que sea"
else
echo "nada de lo anterior"
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 [[ $b -gt $c ]]; then
if [[ $c -gt $d ]]; then
if [[ $d -gt $e ]];then
if [[ $e -gt $f ]]; then
echo "$a es muy grande"
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.
Así la sintasis de los Case
es como puedes ver a continuación,
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)
echo "optimismo, claridad, calor"
;;
naranja)
echo "amigable, social, seguridad"
;;
rojo)
echo "atrevido, excitación, joven"
;;
violeta)
echo "creatividad, imaginativo, sabio"
;;
azul)
echo "útil, fuerza, fiel"
;;
verde)
echo "paz, salud, crecimiento"
;;
blanco)
echo "equilibrio, tranquilidad, neutro"
;;
*)
echo "Lo siento, no conozco ese color"
;;
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"
;;
*)
echo "Ese es un número muy grande"
;;
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"
elif [[ $1 -lt 100 ]]
then
echo "Decenas"
elif [[ $1 -lt 1000 ]]
then
echo "Centenas"
elif [[ $1 -lt 10000 ]]
then
echo "Millares"
else
echo "Un número muy grande"
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
$(($1 <= 10)))
echo "Unidades"
;;
$(($1 <= 100)))
echo "Decenas"
;;
$(($1 <= 1000)))
echo "Centenas"
;;
$(($1 <= 10000)))
echo "Millares"
;;
*)
echo "Un número muy grande"
;;
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
$(($1 <= -125000))*)
periodo="Paleolítico inferior"
;;
$((-125000 < $1 && $1 <= -35000))*)
periodo="Paleolítico medio"
;;
$((-35000 < $1 && $1 <= -10000))*)
periodo="Paleolítico superior"
;;
$((-10000 < $1 && $1 <= -5500))*)
periodo="Mesolítico"
;;
$((-5500 < $1 && $1 <= -4000))*)
periodo="Neolítico"
;;
$((-4000 < $1 && $1 <= -3000))*)
periodo="Edad del Cobre"
;;
$((-3000 < $1 && $1 <= -2000))*)
periodo="Edad del Bronce"
;;
$((-2000 < $1 && $1 <= -1000))*)
periodo="Edad del Hierro"
;;
*)
periodo="Historia"
;;
esac
echo "Te encuentras en el $periodo"
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,
solo una errata, para validar si el archivo existe se una el -f, no -e
Hola Eliot,
Muchas gracias por la corrección, ya está modificado.
Un saludo y gracias.
Saludos, no se que hago mal
Porque me entra en el primer condicional en lugar del segundo?
sma@sma-hpstream11notebookpc:~/Scripts de prueba$ cat nuevaprueba.sh
#! /bin/bash
OPER=»r»
if [[ $OPER==»s» ]]
then
echo oper es s — $OPER
elif [[ $OPER==»r» ]]
then
echo oper es r — $OPER
else
echo oper es distinta —$OPER
fi
echo script finalizado
exit 0
sma@sma-hpstream11notebookpc:~/Scripts de prueba$ bash nuevaprueba.sh
oper es s — r
script finalizado
Hola Danburgues,
Tal como escribes el código, en efecto entras en el primer bucle, pues cuando entras en el primer if,
[[ $OPER==»s» ]] al no poner espacio a ambos lados del ==, el contenido sería [[ «r»==»s» ]] y esto constituye una única expresión de caracteres en lugar de una comparación, lo cual siempre será cierto, pues la expresión no está vacía.
Si ejecutas bash -x nuevaprueba.sh, seguro que lo verás más claro.
La solución es poner espacios entre ==, es decir [[ $OPER == «s» ]] y ya entrará en el segundo if.
Espero que esto te ayude un poco.
Saludos
Hola en el ejercicio if para ver si un numero es par o impar no me funciona me sale como si no leyera el then y pasara directo haciendo que todo numero me de impar, porfa no se si podrías ayudarme con eso hace rato lo estoy intentando soy novato en esto y no he podido hallar el error.
Exelente ing gracias por compartir tu conocimiento
Muchas gracias por la aportación, gracias a esta publicación pude avanzar. Saludos.
hola porfavor tu ayuda es un script que lee los nombres de archivos pdf algunos tiene espacios entre sus nombres y otros tienen el nombre todo junto sin espacios en sus nombres, pero cuando lo entrecomillo igual los toma como si fueran varios archivos,
#!/bin/bash
echo «Este script junta varios pdf en uno»
echo «Ingrese los nombres de los pdf uno por uno»
read -a pdf1
read -a pdf2
read -a pdf3
echo «Ingresa un nombre para del pdf final»
read nombrefinal
echo «Procesando»
echo «!Listoo!»
pdftk «$pdf1» «$pdf2» «$pdf3» cat output «$nombrefinal»
echo «!Listoo!»
hola, como hago para poner simbolos en los case como +,*,/,- ?