Inicio | Pedro Santana |  RSS  Mi tumblelog  Mis photos  Mi CV

 

Archivo de la categoría ‘patrones de diseño’

Patrón singleton en PHP4

Después de mucho tiempo, aquí estoy con la siguiente entrega de la serie sobre patrones de diseño con PHP, la siguiente entrega que estaba planeada era sobre el patrón factory (fábrica o factoría), pero en estos días programé un plugin engine para gelato el cual esta versión aún debe contar con soporte para PHP4 y me topé con que una vez terminado y probado no funcionaba correctamente en PHP4, por dos detalles:

1.- PHP4 no soporta propiedades estáticas (soporta métodos estáticos y variables estáticas dentro de estos métodos, pero NO propiedades),la solución a esto vendrá en otro post.
2.- La implementación del patrón singleton más difundida no funciona como debería.

Veamoslo en bits.

  1. class Singleton{
  2.     var $accion;       
  3.        
  4.     function &instancia(){
  5.         static $_instancia;
  6.         if(!isset($_instancia)){
  7.             $_instancia =& new Singleton();
  8.         }
  9.         return $_instancia;
  10.     }   
  11. }

La clase anterior implementa el código de singleton en PHP4 más encontrado en los ejemplos sobre el tema.

Si la probamos creando el objeto obj1:

  1. $obj1 =& Singleton::instancia();
  2. $obj1->accion = "Imprimir";

Si imprimimos su valor:

  1. var_dump($obj1->accion);
  2. // imprime: string(8) "Imprimir"

Creamos el objeto obj2:

  1. $obj2 =& Singleton::instancia();
  2. $obj2->accion = "Borrar";

¿Qué contenido tiene?

  1. var_dump($obj2->accion);
  2. // imprime: string(6) "Borrar"

Entonces ya tenemos dos objetos derivados de la clase Singleton los cuales al implementar dicho patrón deberamos tener UNA sola instancia, ¿pero esto se cumple?

  1. echo $obj1->accion;

¿Qué nos devuelve?

Imprimir

Como vemos nos regresa el valor “Imprimir” cuando estamos esperando que imprima “Borrar”.

Para solucionarlo basta con tocar una sola línea de código de la clase y cambiar la línea 7 por lo siguiente:

  1. $_instancia = new Singleton();

Ahora si, nuestras pruebas saldrán correctas.

Patrón estrategia

Continuando con la serie de patrones de diseño en PHP5, veamos ahora uno de los patrones que más me llaman la atención: el strategy pattern (patrón estrategia).

Este patrón se usa generalmente cuando se desea cambiar entre diferentes versiones de un algoritmo. Es decir, si tenemos un código que genera un archivo comprimido, bajo ciertas circunstancias, podemos querer un archivo .ZIP y bajo circunstancias diferentes, queremos crear un archivo .TAR.GZ.

Para implementar este patrón debemos declarar una interface (o bien una clase abstracta) base con un método para el algoritmo, el cual es implementado heredando desde las clases concretas.

En el código de nuestro sistema, debemos decidir cual estrategia concreta es la relevante, la cual debe ser instanciada para ser usada.

El ejemplo que usaré para mostrar este patrón, es la selección del idioma del contenido al abrir una página Web. Por simplicidad, asumiré que si la palabra “es” existe en $_SERVER["HTTP_ACCEPT_LANGUAGE"], estamos frente a un usuario que habla español, de otra forma, estamos frente a un usuario que prefiere el inglés.

  1. < ?php
  2. interface langStrategy {
  3.         public function helloWorld();
  4. }
  5.  
  6. class enLang implements langStrategy {
  7.     public function helloWorld() {
  8.         return "Hello World";
  9.     }   
  10. }
  11.  
  12. class esLang implements langStrategy {
  13.     public function helloWorld() {
  14.         return "Hola Mundo";
  15.     }   
  16. }
  17.  
  18. $ln = explode(‘;’,$_SERVER[HTTP_ACCEPT_LANGUAGE]);
  19. if (preg_match(‘/es/’,$ln[0])) {
  20.     $langObj = new esLang();
  21. } else {
  22.     $langObj = new enLang();
  23. }
  24.  
  25. echo $langObj->helloWorld();
  26. ?>

El patrón estrategia es usado a menudo al mismo tiempo que el patrón factory, el cual se encargaría de elegir la estrategia correcta. La siguiente entrega vendrá con este patrón.

Patrón Domain Model

El Domain Model Pattern (algo así como ”Patrón del Modelo de Dominio”), consiste en una clase que representa objetos o responsabilidades particulares en nuestro software.

Por ejemplo, para trabajar con usuarios, cada usuario sería una instancia de una clase “Usuario”.

Normalmente al usar el Domain Model se representan los datos en una o más tablas de una base de datos. Por ejemplo, para la tabla de usuarios de un sistema.

  1. CREATE TABLE `usuarios` (
  2.   `id_usuario` int(10) UNSIGNED NOT NULL AUTO_INCREMENT,
  3.   `nombre` varchar(100) DEFAULT NULL,
  4.   `login` varchar(100) NOT NULL DEFAULT ,
  5.   `password` varchar(64) NOT NULL DEFAULT ,
  6.   `email` varchar(100) DEFAULT NULL,
  7.   PRIMARY KEY  (`id_usuario`)
  8. ) ENGINE = MYISAM ;

Entonces escribimos una clase que represente a cada usuario y sus atributos; es decir el Domain Model:

  1. < ?php
  2. class Usuario {
  3.     public $id_usuario;
  4.     public $nombre;
  5.     public $login;
  6.     public $password;
  7.     public $email;
  8.  
  9.     // metodos para obtener de la BD
  10.     // los datos y asignarlos a las propiedades.
  11. }
  12. ?>

Para usarla debemos instanciarla:

  1. < ?php
  2. $fulano = new Usuario();
  3. echo $fulano->nombre;
  4. ?>

De esta forma dispondremos de las propiedades de los usuarios en nuestros objetos.

Este patrón es el que usamos principalmente en la actual arquitectura de gelato. Pero es importante tener especial cuidado en no caer en el anti-patrón Anemic Domain Model que como todos los anti-patrones hace que se pierdan los beneficios del uso de patrones en nuestro diseño.

Patrón Singleton parte II

Continuamos con la serie de patrones, en esta ocasión vamos a extender la funcionalidad del patrón Singleton para que sea más simple su utilización en nuestros sistemas PHP5.

En el post anterior vimos como implementar el patrón Singleton en nuestras clases. Pero cuando tenemos nuestro sistema con muchas clases que requieren hacer uso de este patrón se vuelve complejo y a la vez más lento el implementar clase por clase todo el código que necesita el patrón para funcionar.

Vamos a mejorar eso utilizando una de las características más importantes de la POO: la herencia.

Paso número 1: Debemos definir la clase base (clase padre); es decir la clase Singleton:

singleton.class.php

  1. < ?php
  2. class singleton {
  3.         private static $instances = array();
  4.        
  5.         public static function getInstance($class) {
  6.                 if (!isset(self::$instances[$class])) {
  7.                         self::$instances[$class] = new $class();
  8.                 }
  9.                 return self::$instances[$class];
  10.         }
  11.        
  12.         private final function __clone() { }
  13. }
  14. ?>

Paso número 2: Crear las clases que vayamos a necesitar extendiendo (heredando) de la clase Singleton.

ejemplo.class.php

  1. < ?php
  2. class ejemplo extends singleton {
  3.         public static function getInstance() {
  4.                 return parent::getInstance(get_class());
  5.         }
  6.        
  7.         public function accion() {
  8.                 echo "Objeto unico";
  9.         }
  10. }
  11. ?>

Paso número 3: Ahora veamos como utilizarlo.

index.php

  1. < ?php
  2. include("singleton.class.php");
  3. include("ejemplo.class.php");
  4.  
  5. $prueba = ejemplo::getInstance();
  6. $prueba->accion();
  7. ?>

De esta forma basta con heredar la clase Singleton en cada una de las clases donde la necesitemos y con hacer polimorfismo del método getInstance() tenemos un Singleton en nuestras clases.

Nos leemos en la siguiente entrega :)

Patrones en PHP5 : Singleton

Como Jorge mencionó en su último post, andamos re-diseñando la arquitectura de gelato cms, para en un par de versiones mudarlo completamente a PHP5 para obtener ventaja de su poderoso motor de OOP y poder hacer un mejor uso de los patrones de diseño.

Por lo que comenzaré con una serie de posts sobre como implementar patrones de diseño con PHP5 y un par de ejemplos un poco más adelante de como aplicarlos en “la vida real” de nuestros proyectos.

El primer patrón con el que iniciaré es el patrón Singleton, el cual considero como la base de la implementación exitosa de muchos patrones.

La teoría nos dice lo siguiente:

El patrón de diseño Singleton (instancia única) está diseñado para restringir la creación de objetos pertenecientes a una clase o el valor de un tipo a un único objeto.

Su intención consiste en garantizar que una clase sólo tenga una instancia y proporcionar un punto de acceso global a ella.

Por lo que al usar el patrón Singleton nos aseguramos que SOLO tendremos una misma instancia de nuestra clase.

¿Cómo hacerlo con PHP5?

  1. < ?php
  2. class Singleton {
  3.  
  4.     // Una propiedad para tener la instancia de la clase
  5.     private static $instancia;   
  6.  
  7.     // Desde aqui se obtiene la instancia OJO con el static
  8.     public static function getInstance () {
  9.         if (!isset(self::$instancia)) {
  10.             $obj = __CLASS__;
  11.             self::$instancia = new $obj;
  12.         }
  13.         return self::$instancia;
  14.     }
  15.  
  16.     // Se crean los metodos requeridos
  17.     public function accion() {
  18.         echo "Objeto unico";
  19.     }   
  20.  
  21.     // Privado se previene la creacion via new
  22.     private function __construct() { }   
  23.  
  24.     // Para evitar la clonacion de este objeto
  25.     private function __clone() {
  26.         throw new Exception(‘No se permite clonar’);
  27.     }
  28. }
  29. ?>

Y para utilizarlo:

  1. < ?php
  2. include("singleton.php");
  3. $prueba = Singleton::getInstance();
  4. $prueba->accion();
  5. ?>

Nos leemos en la siguiente entrega :P