Capítulo 23. Procesamiento de excepciones: ensure

Cuando un método termina su ejecución puede que se necesiten ciertas labores de limpieza. Quizás se tenga que cerrar un fichero abierto, se deban liberar los datos de los buffers, etc. Si siempre hubiese un único punto de salida para cada método, se podría poner con seguridad todo el código de limpieza en un lugar concreto y saber que se ejecutará, sin embargo, un método podría retornar en varios lugares o el código de limpieza podría saltarse inesperadamente debido a una excepción.


begin
  file = open("/tmp/algun_fichero","w")
  # ... Escribimos en el fichero ...
  file.close
end

En el ejemplo superior, si ocurre una excepción durante la parte del código en que se escribe en el fichero, éste quedará abierto. Si no deseamos recurrir al siguiente tipo de redundancia:


begin
  file = open("/tmp/algun_fichero","w")
  # ... Escribimos en el fichero ...
  file.close
rescue
  file.close
  fail # levantamos una excepción
end

Que es desgarbada y se nos escapa de las manos cuando el código se vuelve más complicado debido a que hay que tratar todo return y break.

Por esta razón se añadió otra palabra reservada a esquema begin ... rescue ... end, ensure. El bloque de código de ensure se ejecuta independientemente del éxito o fracaso del bloque de código en begin.


begin
  file = open("/tmp/algun_fichero","w")
  # ... Escribimos en el fichero ...
rescue
  # ... gestionamos las excepciones ...
ensure
  file.close # ... Y esto se ejecuta siempre.
end

Se puede utilizar ensure sin rescue y viceversa, pero si se utilizan en el mismo bloque begin ... end, rescue debe preceder a ensure.