Ir al contenido principal

Programación asíncrona con async/await en .Net

Introducción A partir de la versión 4.5 de .Net framework se ha simplificado de forma considerable la forma en que podemos trabajar con código asíncrono. Con los anteriores frameworks, si queríamos contar con los beneficios de una programación asíncrona nos veíamos obligados a lidiar con una gran complejidad en nuestro código. Esto nos hacía evitar su uso en lo posible a pesar de perder esta importante característica. Stephen Cleary , un MVP especializado en concurrencia define la programación asíncrona como: Una forma de concurrencia que utiliza futuros o "callbacks" para evitar hilos innecesarios. La programación moderna con async y await nos abstrae de la utilización de "callbacks" y nos permite utilizar futuros (Tasks) que se encargarán de notificar al llamante cuando el método asíncrono se complete. async y await Las palabras async y await son las palabras clave que .Net ha introducido en el lenguaje para que podamos implementar métodos asíncronos co...

La importancia del encapsulamiento. Parte 1. Introducción.

Introducción

En programación orientada a objetos el término encapsulamiento es utilizado indistintamente para describir dos conceptos diferentes pero a la vez relacionados entre sí:

  1. Como mecanismo de restricción del acceso a componentes de un objeto;
  2. Como construcción del lenguaje para facilitar el “empaquetado” del estado y el comportamiento.
En este post el significado al que se hace referencia es al primero de ellos. La visibilidad o accesibilidad es el primer paradigma que nos encontramos al empezar a escribir nuestro código. Cuando comenzamos a escribir una clase, interfaz, enumerado o estructura, lo primero que hacemos es establecer su modificador de acceso, y si no lo hacemos, se establecerá el modificador por defecto. Lo mismo ocurre cuando empezamos a escribir cada uno de sus miembros: variables de estado (campos) y métodos: lo primero de todo es escribir su modificador de acceso. A menudo me encuentro con código sin tener en cuenta este aspecto: código demasiado “abierto” al exterior. Los errores más comunes que pueden llevarnos a una mala práctica de encapsulamiento son:
  • ¿Para qué perder el tiempo? Hago todo público y así seguro que compila. Más adelante ya veré si me conviene cambiar el modificador.
  • ¿Y si permito que se lea y escriba en todos mis campos? El código no se rompe y si más adelante necesito acceder ya lo tendré disponible.
  • ¿Qué más da si este método se ve desde fuera o no? A nadie le molesta que haya un método más disponible. De todas formas no creo que sea de mucha utilidad a terceros y seguramente ni lo usen.
  • Está bien, intentaré no abusar del modificador public. Pero… ¿Y si establezco el modificador de todos mis miembros como protegidos? De este modo encapsulo hacia el exterior pero si extiendo la clase no tendré problemas para acceder a los miembros de la base. Al fin y al cabo es su legítimo heredero.
Si sigues alguna de estas prácticas puede que no estés realizando bien el encapsulamiento de tu código. No sólo queremos código que funcione, sino que además sea mantenible y seguro.

Ventajas de encapsular

El encapsulamiento de objetos oculta su interior protegiendo su integridad. Previene que desde fuera se pueda manipular el estado del objeto. Si no se restringe el acceso podría dejar al objeto en un estado inválido o inconsistente.
Otra ventaja del encapsulamiento es la reducción de la complejidad. Una clase debe tener una interfaz pública mínima para su manipulación. Si vamos a usar una clase en la que no se ha restringido de forma eficiente el acceso a sus miembros podemos encontrarnos con un exceso de funcionalidad irrelevante para nosotros, o peor aún, una puerta abierta a miembros que no debería acceder por riesgo a modificar el estado de la clase sin ningún criterio.

Controlando el alcance de nuestros cambios

Una ventaja muy importante subyacente de un buen encapsulamiento es que mejora notablemente nuestra visión del alcance del cambio.
Cuando controlamos de forma eficientemente el encapsulamiento de nuestro código también estamos marcando los límites físicos a los que puede llegar ese código. De esta forma acotamos los “daños colaterales” de futuros cambios.
Por ejemplo, si hacemos un miembro público en una clase publica quiere decir que este miembro puede estar accesible para cualquiera dentro y fuera de nuestro ensamblado. Un cambio de nombre o eliminación de este miembro puede significar que se rompa el código de nuestro ensamblado, algo no muy grave ya que nos daríamos cuenta al compilar y podríamos corregirlo. Pero también podríamos estar rompiendo código de otros ensamblados que dependen del nuestro.
Esto representa un mayor problema porque a veces ni siquiera tenemos acceso a estos ensamblados. La forma correcta de proceder sería mantener esta propiedad “viva” aunque hayamos creado un código alternativo mejor que éste por el mero hecho de mantener la compatibilidad binaria. Podríamos marcar como obsoleto el miembro y esperar a una futura versión para romper esta compatibilidad binaria.

¿Y si este miembro no estuviera debidamente encapsulado?

Imaginemos que este miembro, en realidad, sólo lo hemos utilizado en la propia clase y en dos extensiones de la propia clase, y además la clase podría haber sido internal ya que no se debería utilizar fuera del contexto de nuestro ensamblado. Hubiera sido suficiente un modificador protected para dar visibilidad sólo a la propia clase y a sus extensiones. De esta forma, cuando se deba realizar algún cambio sobre este miembro sabemos con certeza que:
  • no va a romper nada fuera de nuestro ensamblado;
  • podemos revisar fácilmente las dos clases que extienden de la nuestra para ver si afecta nuestro cambio de alguna manera.
Un buen encapsulamiento nos simplifica mucho nuestra tarea y nos da seguridad a la hora de realizar un cambio.
La tranquilidad que da modificar o eliminar un miembro privado de una clase no tiene precio. Sabes que el cambio está afectando sólo a la clase que tienes delante.
Un programador que hace todo público nunca tendrá este tipo de control. El código es mucho menos mantenible y más propenso a errores.

¿Cómo deberíamos proceder?

La regla que yo sigo para encapsular una clase es: asignar la mínima visibilidad posible a los miembros de nuestras clases, tanto campos como métodos. Y si un miembro tiene que ser accesible, deberemos pensar en qué forma y en qué nivel de acceso lo hacemos. Si más adelante necesito ampliar la accesibilidad de un miembro o de un tipo aumentaré su visibilidad hasta el nivel mínimo que cumpla con mis necesidades de acceso.

A nivel de clase procedo de la misma forma: siempre internal y, sólo cuando lo necesito, lo cambio a public.

En definitiva se trata de definir la “interfaz visible” de las clases y sus miembros para que sean accesibles desde otras clases que la usen o desde clases que la extiendan.
Estos son los modificadores utilizados en C# en orden de menor a mayor restricción:

Modificador Accesible desde…
public Sin restricción
 Signal Protected Symbol protected internal El propio ensamblado, la propia clase y sus extensiones
 Signal Friend/Internal Symbol internal El propio ensamblado y la propia clase
 Signal Protected Symbol protected La propia clase y sus extensiones
 Signal Private Symbol private La propia clase

>> Ir a Parte 2. Un ejemplo de encapsulamiento en la vida real.

Comentarios

Publicar un comentario

Entradas populares de este blog

La importancia del encapsulamiento. Parte 2. Un ejemplo de encapsulamiento en la vida real.

Un ejemplo de encapsulamiento en la vida real Supongamos un taller de carpintería que fabrica muebles a medida. Por una parte tendríamos el propio taller, que sería la clase a encapsular. Por otra los clientes, que serían nuestros usuarios y representarían las clases que USAN la clase taller. Y por otra parte podríamos pensar en talleres filiales que representarían una especialización del taller matriz, es decir, clases que extienden al taller de carpintería y por lo tanto SON talleres. El taller cuenta con funcionalidades puestas al servicio de los clientes como la elaboración de presupuestos, consulta de catálogos, selección de tipo de madera, selección de color, etc. También cuenta con funcionalidades propias del taller como cortar, lijar, pintar, barnizar, ensamblar mueble, etc. Para la elaboración de los presupuestos el taller matriz cuenta con una tarifa de precios que sólo los talleres pueden acceder para poder elaborar los presupuestos. Esta tarifa la fija el taller mat...

Programación asíncrona con async/await en .Net

Introducción A partir de la versión 4.5 de .Net framework se ha simplificado de forma considerable la forma en que podemos trabajar con código asíncrono. Con los anteriores frameworks, si queríamos contar con los beneficios de una programación asíncrona nos veíamos obligados a lidiar con una gran complejidad en nuestro código. Esto nos hacía evitar su uso en lo posible a pesar de perder esta importante característica. Stephen Cleary , un MVP especializado en concurrencia define la programación asíncrona como: Una forma de concurrencia que utiliza futuros o "callbacks" para evitar hilos innecesarios. La programación moderna con async y await nos abstrae de la utilización de "callbacks" y nos permite utilizar futuros (Tasks) que se encargarán de notificar al llamante cuando el método asíncrono se complete. async y await Las palabras async y await son las palabras clave que .Net ha introducido en el lenguaje para que podamos implementar métodos asíncronos co...