Capítulo 9: Usando Funciones en Bash

En esta lección, aprenderás a crear funciones, devolver valores de funciones y pasar argumentos de funciones en scripts de shell bash.
Warp Terminal

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.

Automatizando con Bash
En el último capítulo, te mostraré algunos scripts de automatización que puedes ampliar más tarde para automatizar cualquier tarea que desees.
About the author
Marco Carmona

Marco Carmona

My name is Marco Antonio Carmona, I'm a physics and data science student, a great and passionate reader, and randomly, my favorite hobby is writing about what I learn day by day.

It's FOSS

Making You a Better Linux User

itsfoss happy penguin

Great! You’ve successfully signed up.

Welcome back! You've successfully signed in.

You've successfully subscribed to It's FOSS.

Success! Check your email for magic link to sign-in.

Success! Your billing info has been updated.

Your billing was not updated.