DEV Community

Fabio Sierro Cartolano
Fabio Sierro Cartolano

Posted on

ORMs: ¿solución o problemas en puerta?

Una reflexión desde la experiencia práctica sobre una decisión que muchos equipos toman sin medir sus consecuencias.

Hay decisiones tecnológicas que se toman de manera casi automática, sin demasiado debate y con la convicción de que son la opción moderna y correcta. El uso de ORMs —Object-Relational Mappers, herramientas como Entity Framework en .NET o Hibernate en Java— es una de ellas. Esta no es una crítica a los ORMs como herramienta; son, en muchos contextos, soluciones brillantes. El problema surge cuando se los adopta como arquitectura base de acceso a datos en sistemas complejos, de gran escala, sin analizar seriamente las consecuencias.

El ORM como decisión por defecto
En la mayoría de los equipos medianos, la elección del ORM no es producto de un análisis arquitectónico profundo. Es el resultado de la presión por entregar rápido y la familiaridad de los desarrolladores con la herramienta. El ORM reduce la fricción inicial de manera notable: permite interactuar con la base de datos sin escribir SQL y ver resultados en minutos. Eso tiene valor. El problema es que esa facilidad oscurece una complejidad que no desaparece, simplemente se difiere. Y cuando aparece —y siempre aparece— lo hace cuando el sistema ya tiene usuarios, los datos ya son muchos y nadie tiene tiempo de refactorizar.

El problema de visibilidad compartida
En sistemas de cierta escala, el equipo de datos ve planes de ejecución, índices y consultas lentas, pero no el código que las genera. El desarrollador ve modelos de objetos y LINQ, pero no lo que el ORM produce realmente ni cómo impacta en el motor. En el medio está el ORM, actuando como caja negra que traduce de un mundo al otro sin que ninguno de los dos equipos tenga control real sobre esa traducción.
El resultado es predecible: el equipo de datos pasa sus días agregando índices y ajustando estadísticas sin atacar la raíz, que muchas veces es una consulta mal construida por el ORM que ningún índice va a salvar. Un índice puede mejorar una consulta ineficiente, pero no puede convertirla en una consulta bien diseñada.

El argumento de la portabilidad y otros mitos
El argumento de la portabilidad —poder cambiar el motor de base de datos sin tocar el código— suena razonable en teoría. En la práctica, en sistemas maduros con años de historia, es casi una fantasía. Sacrificar performance cotidiana para estar preparado para algo que probablemente nunca ocurra no es arquitectura prudente: es arquitectura ansiosa. El argumento del versionado tampoco se sostiene: con herramientas como los Database Projects de SSDT, Flyway o Liquibase, los objetos de base de datos viven en el repositorio con historial completo e integración CI/CD. En 2026, ese argumento es técnicamente débil.

La separación de responsabilidades y los procedimientos almacenados
El argumento más sólido contra los stored procedures es el de la separación de responsabilidades: si un SP contiene lógica de negocio, algo que pertenece a la capa de aplicación está viviendo en la base de datos. Eso es un problema real. Pero hay una distinción que frecuentemente se ignora: un SP que decide si un cliente recibe un descuento mezcla responsabilidades; un SP que ejecuta un join complejo sobre millones de registros está haciendo exactamente lo que corresponde, cerca del motor que almacena los datos. El problema no es la herramienta sino cómo se usa. Y la diferencia clave es que un SP mal escrito es un problema conocido y localizado; un ORM mal configurado produce problemas difusos y emergentes que aparecen de formas inesperadas a medida que el sistema escala.

La agilidad operacional que nadie menciona
Hay un argumento que raramente aparece en los debates teóricos sobre ORMs, pero que cualquier persona con experiencia práctica en sistemas en producción reconoce de inmediato: la agilidad para responder a los cambios cotidianos que piden los clientes.
Los clientes siempre piden más información. Es una constante universal del desarrollo de software. 'Necesito ver también la fecha', 'agreguen el nombre del responsable', 'quiero poder filtrar por región'. Pedidos simples, frecuentes, inevitables. Y aquí es donde la diferencia entre un SP y un ORM se vuelve muy concreta.
Con un procedimiento almacenado, ese ciclo es notablemente corto: el arquitecto o administrador de datos agrega el campo a la consulta, y en minutos el cambio está disponible. Sin tocar código de aplicación, sin recompilar, sin coordinar con el equipo de frontend, sin un pipeline de CI/CD completo y sin una ventana de mantenimiento con su riesgo asociado. Un campo que ya existía en la base de datos simplemente empieza a mostrarse.
Esto es posible, en buena medida, gracias al avance de los componentes dinámicos en el desarrollo de aplicaciones modernas. Una grilla de datos —uno de los componentes más ubicuos en sistemas de gestión empresarial— puede programarse para mutar automáticamente de acuerdo a los campos que recibe: si el backend devuelve una columna nueva, el componente la detecta, la incorpora y le aplica de forma automática las mismas propiedades de presentación, formato y comportamiento que ya tienen las columnas existentes. Ordenamiento, filtrado, ancho adaptativo, alineación: todo se hereda sin intervención manual. El frontend no necesita saber de antemano qué campos va a mostrar. Simplemente los muestra.
Con un ORM, ese mismo cambio recorre un camino considerablemente más largo: agregar la propiedad al modelo o DTO, ajustar el mapeo, modificar la query LINQ si corresponde, compilar, testear, hacer el commit, el pull request, la revisión de código, el pipeline de build y finalmente el deploy. Todo eso para un campo que ya existía en la base de datos.
La ironía es llamativa: el ORM se adopta argumentando velocidad y productividad, pero en la fase donde más se necesita agilidad de respuesta —el mantenimiento y la evolución continua del sistema con usuarios reales— el procedimiento almacenado resulta significativamente más ágil para exactamente el tipo de cambios que con mayor frecuencia solicitan los clientes. Los sistemas viven mucho más tiempo en modo evolución que en modo construcción, y esa realidad rara vez se considera al momento de elegir la arquitectura de acceso a datos.

Una solución que preserva la arquitectura: DTOs de Response independientes
La descripción anterior podría generar un interrogante legítimo en cualquier arquitecto que trabaje con diseños modernos: si el componente del frontend adapta la grilla automáticamente a los campos que recibe, ¿qué rol cumplen los DTOs? ¿No estamos resignando uno de los pilares de una arquitectura sólida —el contrato explícito entre capas— en nombre de la agilidad operacional? La tensión es real y vale la pena resolverla con precisión.
La respuesta no es eliminar los DTOs sino ubicarlos correctamente. La solución consiste en crear un proyecto independiente dedicado exclusivamente a los DTOs de Response, separado de la capa de datos, de la capa de negocio y de cualquier otra dependencia. Este proyecto actúa como contrato puro entre el backend y el frontend: define qué forma tienen los datos que el sistema expone, sin mezclar esa responsabilidad con ninguna otra.
Bajo este esquema, cuando un cliente solicita un campo adicional, el cambio es mínimo y perfectamente localizado: se agrega el campo en el procedimiento almacenado y se agrega la propiedad correspondiente en el DTO de Response de ese SP específico. Nada más. No se toca lógica de negocio, no se modifican otros DTOs, no se altera ninguna otra parte del sistema. El componente dinámico del frontend recibe el contrato actualizado y adapta la presentación de forma automática. El ciclo es corto, el impacto es quirúrgico y la arquitectura permanece intacta.
Este enfoque tiene además beneficios que van más allá de la agilidad inmediata. Al ser un proyecto independiente, los DTOs de Response no generan dependencias circulares con ninguna capa. Los cambios quedan explícitos y trazables en el repositorio, asociados al SP correspondiente. Y si en el futuro se necesita versionar una API —algo habitual en sistemas que crecen y evolucionan— la estructura ya está preparada para hacerlo de forma ordenada, sin romper consumidores existentes.
El único riesgo, como ocurre con toda decisión arquitectónica, es de disciplina: la tentación de reutilizar un DTO de Response para otro propósito, o de mapear directamente desde una entidad de datos saltándose el contrato, existe y se vuelve más tentadora bajo presión. Pero ese riesgo no es un argumento contra la solución; es simplemente un recordatorio de que toda arquitectura sólida requiere governance sostenido en el tiempo. La propuesta técnica es correcta. Lo que la sostiene o la degrada es la cultura del equipo que la adopta.

Por qué los referentes técnicos no siempre ayudan
Los desarrolladores que publican contenido técnico son, en su mayoría, expertos en código, patrones y frameworks, no necesariamente en arquitectura de datos. Cuando alguien muy bueno en su área opina sobre un área adyacente, lo hace con puntos ciegos. El problema es que la audiencia no siempre distingue entre 'es muy bueno en desarrollo' y 'es una autoridad en todo lo que el desarrollo toca'. Hay además un componente estético: una consulta LINQ se ve elegante en pantalla; un stored procedure, no. En el mundo del contenido técnico, eso influye más de lo que nos gusta admitir.

Lo que hacen los sistemas serios
Empresas como Amazon, Google o Netflix no usan ORMs para sus sistemas críticos. Trabajan con acceso a datos muy controlado y consultas nativas, porque la escala no les permite el lujo de una capa de abstracción opaca. Los sistemas bancarios y financieros de primer nivel siguen usando procedimientos almacenados extensivamente. No por conservadurismo irracional, sino porque necesitan trazabilidad, control absoluto y performance predecible. Un banco no puede permitirse que una consulta generada por un ORM cambie su plan de ejecución porque se actualizó una versión del framework. No es conservadurismo irracional; es la experiencia acumulada de sistemas que no pueden fallar, escrita en decisiones arquitectónicas concretas.

Entonces, ¿cuál es el lugar del ORM?
El ORM tiene un lugar legítimo: operaciones CRUD simples, equipos pequeños, prototipos, sistemas donde la escala no es el problema central. El problema no es la herramienta sino la ausencia de criterio sobre cuándo usarla y cuándo no. En sistemas complejos con equipos diferenciados y grandes volúmenes de datos, la arquitectura de acceso a datos debería ser una decisión explícita, no el resultado de tomar el camino de menor resistencia.
Ted Neward, arquitecto con más de treinta años de experiencia, lo describió hace casi dos décadas con una analogía que sigue siendo vigente: el ORM es el Vietnam de las ciencias de la computación. Empieza bien, se complica con el tiempo, y antes de que te des cuenta estás atrapado en un compromiso sin salida clara. La experiencia práctica —no los tutoriales de YouTube— tiende a llegar a las mismas conclusiones. Y eso, en sí mismo, dice bastante.

— Reflexión basada en experiencia práctica en desarrollo y arquitectura de software.

Top comments (0)