Patrón Prototype
Crea nuevos objetos copiando los existentes. Un patrón de diseño creacional para clonación y duplicación de objetos.
Patrón Prototype
Visión General
El Patrón Prototype es un patrón de diseño creacional que crea nuevos objetos copiando los existentes. En lugar de construir objetos desde cero usando constructores, clonas una instancia prototipo y opcionalmente la personalizas. Esto es especialmente útil cuando la creación de objetos es costosa, involucra configuraciones complejas, o cuando necesitas objetos similares pero no idénticos.
Cuándo Usarlo
Usa el Patrón Prototype cuando:
- La creación de objetos es costosa o involucra inicialización compleja
- Necesitas muchos objetos que difieren ligeramente entre sí
- Quieres evitar subclasificar solo para variar configuraciones
- Los objetos tienen numerosos estados y combinaciones posibles
- Quieres preservar el estado de un objeto existente como punto de partida
Solución
Python
import copy
from abc import ABC, abstractmethod
class Document(ABC):
def __init__(self, content="", formatting=None):
self.content = content
self.formatting = formatting or {}
@abstractmethod
def clone(self):
pass
def __str__(self):
return f"Document(content={self.content}, formatting={self.formatting})"
class Report(Document):
def __init__(self, content="", formatting=None, sections=None):
super().__init__(content, formatting)
self.sections = sections or []
def clone(self):
# Deep copy asegura que los objetos anidados sean independientes
return copy.deepcopy(self)
def __str__(self):
return f"Report(content={self.content}, sections={self.sections})"
# Crear un report prototipo con secciones estándar
prototype = Report(
content="Plantilla de Informe Anual",
formatting={"font": "Arial", "size": 12},
sections=["Introducción", "Finanzas", "Conclusión"]
)
# Clonar y personalizar
report_a = prototype.clone()
report_a.content = "Informe Anual 2024"
report_a.sections.append("Apéndice")
report_b = prototype.clone()
report_b.content = "Informe Anual 2023"
report_b.formatting["size"] = 14
print(report_a)
print(report_b)
print(report_a.formatting is report_b.formatting) # False (deep copy)
JavaScript
class Document {
constructor(content = "", formatting = {}) {
this.content = content;
this.formatting = formatting;
}
clone() {
// Deep clone usando structuredClone (navegadores modernos/Node 17+)
return structuredClone(this);
}
}
class Report extends Document {
constructor(content = "", formatting = {}, sections = []) {
super(content, formatting);
this.sections = sections;
}
clone() {
return structuredClone(this);
}
}
// Crear un prototipo
const prototype = new Report(
"Plantilla de Informe Anual",
{ font: "Arial", size: 12 },
["Introducción", "Finanzas", "Conclusión"]
);
// Clonar y personalizar
const reportA = prototype.clone();
reportA.content = "Informe Anual 2024";
reportA.sections.push("Apéndice");
const reportB = prototype.clone();
reportB.content = "Informe Anual 2023";
reportB.formatting.size = 14;
console.log(reportA.formatting === reportB.formatting); // false
Java
public interface Prototype {
Prototype clone();
}
public class Report implements Prototype {
private String content;
private java.util.Map<String, Object> formatting;
private java.util.List<String> sections;
public Report(String content, java.util.Map<String, Object> formatting,
java.util.List<String> sections) {
this.content = content;
this.formatting = new java.util.HashMap<>(formatting);
this.sections = new java.util.ArrayList<>(sections);
}
@Override
public Report clone() {
return new Report(
this.content,
new java.util.HashMap<>(this.formatting),
new java.util.ArrayList<>(this.sections)
);
}
public void setContent(String content) { this.content = content; }
@Override
public String toString() {
return "Report{content='" + content + "', sections=" + sections + "}";
}
}
// Uso
Report prototype = new Report(
"Plantilla de Informe Anual",
java.util.Map.of("font", "Arial", "size", 12),
java.util.List.of("Introducción", "Finanzas", "Conclusión")
);
Report reportA = prototype.clone();
reportA.setContent("Informe Anual 2024");
Report reportB = prototype.clone();
reportB.setContent("Informe Anual 2023");
Explicación
El Patrón Prototype tiene dos roles clave:
- Interfaz Prototype — declara un método
clone()que todos los objetos clonables implementan - Prototipos Concretos — implementan el método clone, produciendo copias exactas de sí mismos
El desafío clave es decidir entre copia superficial (copia referencias) y copia profunda (copia objetos referenciados). Usa copia superficial para objetos simples, copia profunda cuando los objetos anidados deben permanecer independientes.
Variantes
| Variante | Caso de Uso | Compromiso |
|---|---|---|
| Clon Superficial | Objetos simples sin mutables anidados | Rápido, pero referencias compartidas pueden causar efectos secundarios |
| Clon Profundo | Objetos complejos con estado anidado | Más lento, pero copias completamente independientes |
| Basado en Registro | Múltiples prototipos identificados por clave | Búsqueda flexible, agrega gestión de registro |
| Clon por Serialización | Copia profunda vía serializar/deserializar | Maneja grafos complejos, pero más lento |
Buenas Prácticas
- Usa copia profunda para objetos mutables anidados para prevenir estado compartido no deseado entre clones
- Implementa
clone()explícitamente en lugar de confiar en el comportamiento por defecto del lenguaje - Documenta si el clone es superficial o profundo para que los llamadores sepan qué esperar
- Considera la inmutabilidad — los objetos inmutables no necesitan clonación
- Agrega un registro de prototipos cuando tengas múltiples instancias prototipo y necesites recuperarlas por nombre o tipo
Errores Comunes
- Usar copia superficial en objetos con estado mutable anidado, causando que los clones se afecten mutuamente
- Confiar en
Object.clone()por defecto en Java sin manejar la copia profunda de campos mutables - Olvidar que
Object.assigny el spread operator de JavaScript realizan copias superficiales - Clonar objetos que contienen recursos externos (manejadores de archivos, sockets) sin reinicializar esos recursos
- No manejar referencias circulares durante la clonación profunda, llevando a recursión infinita
Preguntas Frecuentes
P: ¿Cuál es la diferencia entre Prototype y Factory? R: Factory crea objetos usando un método/clase separado. Prototype crea objetos copiando una instancia existente. Usa Factory cuando la lógica de creación es compleja; usa Prototype cuando los objetos son costosos de construir desde cero.
P: ¿Debo usar siempre copia profunda? R: No. Usa copia superficial cuando los objetos anidados son inmutables o cuando quieres referencias compartidas intencionalmente. Usa copia profunda cuando los objetos anidados son mutables y deben ser independientes.
P: ¿Cómo se compara Prototype con el patrón Builder? R: Builder construye un objeto paso a paso. Prototype copia un objeto completamente construido. Resuelven problemas diferentes: Builder es para construcción compleja, Prototype es para duplicación.