Cuando tus scripts de bash se hace cada vez más grandes, ¡las cosas pueden volverse un desorden!
Es posible que te encuentres reescribiendo los mismos trozos de código una y otra vez en diferentes partes de tus scripts bash.
Por suerte, puedes evitar reescribir el código utilizando funciones en bash que harán que tus scripts sean más organizados y legibles.
En esta lección, aprenderás a crear funciones, devolver valores de funciones y pasar argumentos de funciones en scripts de shell bash.
Además, aprenderá cómo funciona el ámbito de las variables y cómo definir funciones recursivas.
Creación de funciones en bash
Hay dos sintaxis diferentes para declarar funciones bash. La siguiente sintaxis es la más utilizada para crear funciones bash:
nombre_de_funcion () {
comandos
}
La segunda forma menos utilizada de crear funciones bash comienza con la función de trabajo reservada seguida del nombre de la función como sigue:
function nombre_de_funcion {
comandos
}
Ahora bien, hay un par de cosas que debes tener en cuenta cuando trabajes con funciones:
- Una función nunca se ejecutará/ejecutará a menos que se invoque/llame a la función.
- La definición de la función debe preceder a cualquier llamada a la función.
Cada vez que quieras que se ejecute una función, sólo tienes que llamarla. Una llamada a una función se realiza simplemente haciendo referencia al nombre de la función.
Echa un vistazo al siguiente script bash fun.sh:
#!/bin/bash
hola () {
echo "Hola Mundo"
}
hola
hola
hola
Definí una función llamada hola que simplemente hace eco de la línea “Hola Mundo” en la terminal. Fíjate que hice tres llamadas a la función hola y, por lo tanto, si ejecutas el script, verás la línea “Hola Mundo” impresa tres veces en la pantalla:
team@itsfoss:~$ ./fun.sh
Hola Mundo
Hola Mundo
Hola Mundo
Devolución de valores de funciones en bash
En muchos lenguajes de programación, las funciones devuelven un valor cuando son llamadas; sin embargo, este no es el caso de bash ya que las funciones de bash no devuelven valores.
Cuando una función bash termina de ejecutarse, devuelve el estado de salida del último comando ejecutado capturado en la variable $?. El cero indica que la ejecución ha sido exitosa o un entero positivo distinto de cero (1-255) para indicar que ha fallado.
Puedes utilizar una sentencia return para modificar el estado de salida de la función. Por ejemplo, eche un vistazo al siguiente script error.sh:
#! /bin/bash
error () {
blabla
return 0
}
error
echo "El estado return de la función error es: $?"
Si ejecutas el script bash error.sh, podrías sorprenderte de la salida:
team@itsfoss:~$ ./error.sh
./error.sh: line 4: blabla: command not found
El estado return de la función error es: 0
Sin la declaración return 0, la función de error nunca habría devuelto un estado de salida distinto de cero, ya que blabla resulta en un error de comando no encontrado.
Así que como puedes ver, aunque las funciones de bash no devuelven valores, hice una solución alterando los estados de salida de las funciones.
También debes saber que una sentencia return termina inmediatamente una función.
Pasar argumentos a la función bash
Puedes pasar argumentos a una función igual que puedes pasar argumentos a un script de bash. Sólo tienes que incluir los argumentos cuando haces la llamada a la función.
Para demostrarlo, echemos un vistazo al siguiente script bash espar.sh:
#!/bin/bash
espar () {
if [ $(($1 % 2)) -eq 0 ]; then
echo "$1 es par."
else
echo "$1 es impar."
fi
}
espar 3
espar 4
espar 20
espar 111
La función espar() comprueba si un número es par o impar. Hice cuatro llamadas a la función espar(). Para cada llamada a la función, suministré un número que es el primer aumento de la función iseven() y es referenciado por la variable $1 en la definición de la función.
Vamos a ejecutar el script bash espar.sh para asegurarnos de que funciona:
team@itsfoss:~$ ./espar.sh
3 es impar.
4 es par.
20 es par.
111 es impar.
También debes saber que los argumentos de las funciones de bash y los argumentos de los scripts de bash son dos cosas diferentes. Para contrastar la diferencia, echa un vistazo al siguiente script bash funarg.sh:
#!/bin/bash
fun () {
echo "$1 es el primer argumento de fun()"
echo "$2 es el segundo argumento de fun()"
}
echo "$1 es el primer argumento del script."
echo " $2 es el segundo argumento del script."
fun Yes 7
Ejecuta el script con un par de argumentos y observa el resultado:
team@itsfoss:~$ ./funarg.sh Cool Cosa
Cool es el primer argumento del script.
Cosa es el segundo argumento del script.
Yes es el primer argumento de fun()
7 es el segundo argumento de fun()
Como puede ver, aunque has utilizado las mismas variables $1 y $2 para referirse tanto a los argumentos del script como a los de la función, producen resultados diferentes cuando se llaman desde una función.
Variables locales y globales en las funciones de bash
Las variables de Bash pueden tener un alcance global o local. Puedes acceder a una variable global en cualquier parte de un script bash sin importar el ámbito. Por el contrario, sólo se puede acceder a una variable local desde la definición de su función.
Para demostrarlo, echa un vistazo al siguiente script bash scope.sh:
#!/bin/bash
v1='A'
v2='B'
myfun() {
local v1='C'
v2='D'
echo "Dentro de myfun(): v1: $v1, v2: $v2"
}
echo "Antes de llamar a myfun(): v1: $v1, v2: $v2"
myfun
echo "Después de llamar a myfun(): v1: $v1, v2: $v2"
Primero definí dos variables globales v1 y v2. A continuación, dentro de la definición de myfun(), utilicé la palabra clave local para definir una variable local v1 y modifiqué la variable global v2. Ten en cuenta que puedes utilizar el mismo nombre de variable para las variables locales en diferentes funciones.
Ahora vamos a ejecutar el script:
team@itsfoss:~$ ./scope.sh
Antes de llamar a myfun(): v1: A, v2: B
Dentro de myfun(): v1: C, v2: D
Después de llamar a myfun(): v1: A, v2: D
De la salida del script, se puede concluir lo siguiente:
- Una variable local que tenga el mismo nombre que una variable global tendrá prioridad sobre las variables globales dentro del cuerpo de una función.
- Puedes cambiar una variable global desde dentro de una función.
Funciones recursivas
Una función recursiva es una función que se llama a sí misma. Las funciones recursivas resultan útiles cuando se intenta resolver un problema de programación que puede dividirse en subproblemas más pequeños.
La función factorial es un ejemplo clásico de función recursiva. Echa un vistazo al siguiente script bash factorial.sh:
#!/bin/bash
factorial () {
if [ $1 -le 1 ]; then
echo 1
else
last=$(factorial $(( $1 -1)))
echo $(( $1 * last ))
fi
}
echo -n "4! es: "
factorial 4
echo -n "5! es: "
factorial 5
echo -n "6! es: "
factorial 6
Toda función recursiva debe comenzar con un caso base que es necesariamente para terminar la cadena de llamadas a funciones recursivas. En la función factorial(), el caso base se define como sigue:
if [ $1 -le 1 ]; then
echo 1
Ahora deduce el caso recursivo para la función factorial. Para calcular el factorial de un número n donde n es un número positivo mayor que uno, puedes multiplicar n por el factorial de n-1:
factorial(n) = n * factorial(n-1)
Utilicemos la ecuación anterior para escribir este caso recursivo:
last=$(factorial $(( $1 -1)))
echo $(( $1 * last ))
Ahora ejecuta el script y asegúrate de obtener los resultados correctos:
team@itsfoss:~$ ./factorial.sh
4! es: 24
5! es: 120
6! es: 720
Como ejercicio adicional, intenta escribir una función recursiva para calcular el enésimo número de Fibonacci. Primero, trata de llegar al caso base y luego al caso recursivo; ¡lo has conseguido!
¡Espectacular! Con esto llegamos al final de esta lección. ¡Espero que hayas disfrutado creando funciones en bash! En la próxima y última lección de este curso, aplicarás todo lo que has aprendido hasta ahora para escribir efectivos scripts en bash que automaticen aburridas tareas administrativas.