Translate

sábado, 2 de mayo de 2026

Records vs Project Valhalla en Java


Java viene evolucionando fuerte en dos direcciones:

  • Java Records → escribir menos código
  • Project Valhalla → ejecutar más rápido


A primera vista parecen competir… pero en realidad se complementan.

  • Usá Records → claridad, simplicidad, APIs
  • Usá Valhalla (Value Types) → rendimiento, memoria
  • Usá ambos → lo ideal en sistemas complejos


Records: el rey de la simplicidad

record User(String name, int age) {}


✔ Inmutables

✔ Menos boilerplate

✔ equals, hashCode, toString automáticos


¿Cuándo usar Records?

DTOs / APIs


record ProductDTO(String name, double price) {}


Perfecto para:

  • REST APIs
  • serialización JSON
  • comunicación entre capas


Modelos simples

record Money(double amount, String currency) {}


Cuando te importa más la legibilidad que el rendimiento


Lógica de negocio

  • servicios
  • respuestas intermedias


El overhead de objetos no suele ser un problema


Limitación clave, un record:

  • vive en el heap
  • tiene identidad
  • usa referencias


Valhalla: el rey del rendimiento

Ejemplo conceptual:


value class Point {

    int x, y;

}



✔ Sin identidad

✔ Sin overhead de objeto

✔ Datos contiguos en memoria


¿Cuándo usar Value Types?

Grandes volúmenes de datos


value class Point {

    int x, y;

}


List<Point>


Mucho más eficiente que objetos tradicionales


Simulaciones

  • físicas
  • partículas
  • coordenadas


Miles o millones de instancias


Alto rendimiento

  • trading
  • analytics
  • cálculos intensivos


Menos GC = mejor performance


¿Qué cambia realmente?

Sin Valhalla:

List<Point> // lista de referencias


Con Valhalla:

[x1, y1, x2, y2, x3, y3]


Datos planos → mejor cache → más velocidad


Cuándo NO usar Valhalla

  • Entidades (`User`, `Order`)
  • Objetos con identidad
  • Integración con frameworks clásicos


Si necesitás identidad → no es value type

Lo mejor: combinarlos

Acá está la clave real 👇


Caso 1: Record + Value Type


value class Point {

    int x, y;

}


record Circle(Point center, double radius) {}


Tenés:

API clara (record)

datos eficientes (value)


Caso 2: separación por capas


// API

record PointDTO(int x, int y) {}


// Core

value class Point {

    int x, y;

}


Separás:

  • interfaz → legible
  • core → performante


Caso 3: colecciones grandes


value class Price {

    double amount;

}


List<Price>


Sin boxing → sin referencias → 🔥


Futuro: value records

Se viene algo así:

value record Point(int x, int y) {}


Combina:

  • simplicidad (record)
  • eficiencia (value type)


Conclusión, no es una pelea, es una estrategia:

  • Records → hacen feliz al desarrollador
  • Valhalla → hace feliz a la JVM



Java vs C#: Records y Value Types vs Records y Structs


Durante años, C# llevó ventaja en expresividad y control de memoria.

Pero con Java Records y Project Valhalla, Java está alcanzando… y en algunos aspectos, intentando superarlo.


👉 La pregunta es:

 ¿Java está copiando a C#… o lo está mejorando?


Empecemos por records: empate técnico

En C#

record Person(string Name);


En Java

record Person(String name) {}


✔ Inmutables

✔ Comparación por valor

✔ Menos boilerplate


Conclusión: Empate total — ambos lenguajes convergieron al mismo concepto


Value Types vs Structs

En C#: struct


struct Point {

    public int X;

    public int Y;

}


✔ Tipo por valor

✔ Generalmente en stack

✔ Muy eficiente


Problemas:

Boxing (se convierte en objeto)

Copias implícitas

Comportamiento a veces confuso


 En Java: Value Types (Valhalla)


value class Point {

    int x, y;

}


✔ Sin identidad

✔ Sin overhead de objeto

✔ Puede ser “flattened”


Y lo más importante: No cambia su naturaleza según el contexto


Diferencia clave

C# → tipo híbrido (a veces valor, a veces objeto)

Java → tipo consistente (siempre value type)


Punto para Java (a nivel diseño)


Memoria y layout

  • Stack vs Heap
  • Depende del contexto
  • Puede generar copias innecesarias


 Java (Valhalla)

  • Flattening automático
  • Mejor locality
  • Optimización por la JVM


Ejemplo mental:

En java esto: 

List<Point>


Podría ser:

[x1, y1, x2, y2, x3, y3]


Mucho más eficiente que referencias



Generics (el golpe fuerte de Java)


C#

List<int> 


Soportado desde hace años


Java (hoy)

List<Integer>  (boxing)


Java (con Valhalla)

List<int> ✔

List<Point> ✔


Sin boxing y sin referencias


Este es uno de los mayores avances de Java


Complejidad del modelo mental


En C#

  • class vs struct
  • boxing/unboxing
  • copias implícitas
  • Puede ser confuso


En Java

  • class vs value class
  • sin identidad vs con identidad
  • Más simple conceptualmente


Ecosistema y madurez

C#

✔ Ya está en producción

✔ Frameworks adaptados

✔ Casos reales


 Java (Valhalla)

⚠️ En desarrollo

⚠️ APIs en evolución

⚠️ Falta adopción


Hoy no podés usarlo en producción estándar



¿Quién lo hizo mejor?


Hoy: C#

Porque:

  • ya funciona
  • es estable
  • está probado


 A futuro: Java (posiblemente)

Porque:

  • evita errores históricos (boxing, copias)
  • diseño más consistente
  • mejor integración con la JVM



HTTP 418: I'm a teapot


Si alguna vez viste el código HTTP 418, probablemente pensaste: “¿Esto es real o me están trolleando?”

Spoiler: ambas cosas.


¿Qué significa el error 418?

El código HTTP 418 – "I'm a teapot" es un código de error del cliente (4xx) que, literalmente, significa:

 “Soy una tetera y no puedo hacer café” 


La idea es simple:

El cliente hace una request (ej: “haceme café” ☕)

El servidor responde:

  “No puedo, soy una tetera”


Este código nace en 1998 en el RFC 2324, llamado:

Hyper Text Coffee Pot Control Protocol (HTCPCP)


Sí, un protocolo para controlar cafeteras por internet… y sí, fue publicado el 1 de abril (April Fools). 


La especificación decía cosas como:

  • Método HTTP: BREW
  • Headers para azúcar o leche
  • Y este glorioso error 418

 ¿Es un chiste o es real?

Las dos:

  • Es parte “oficial” del ecosistema HTTP (aunque reservado/no usado formalmente) 
  • No tiene uso serio en producción
  • Pero muchos frameworks lo incluyen igual


Ejemplos reales:

  • Java / Spring: podés devolver 418
  • .NET: `StatusCodes.Status418ImATeapot`
  • Node / Go / Python también lo soportan 


¿Se usa en la vida real?

No debería… pero igual aparece:

  • Easter eggs en APIs
  • Testing de errores
  • Bloqueo de bots
  • Mensajes humorísticos para developers


Incluso hay devs que lo usan para decir: “Tu request es absurda”


Dejo link: https://developer.mozilla.org/en-US/docs/Web/HTTP/Reference/Status/418

Proyecto Valhalla en Java: Tipos de Valor y el Futuro del Rendimiento


Cuando hablamos de la evolución de Java, uno de los proyectos más interesantes (y menos comprendidos) es Project Valhalla.


Si alguna vez sentiste que Java es “pesado” por culpa de los objetos… este proyecto apunta directamente a eso.

Project Valhalla es una iniciativa dentro de OpenJDK cuyo objetivo principal es introducir:

Tipos de valor (Value Types / Inline Classes)

Esto permite representar datos sin el costo de los objetos tradicionales.

En Java, casi todo es un objeto… y eso tiene consecuencias:


Integer x = 10;


Esto implica:

  • Heap allocation
  • Indirección (punteros)
  • Presión sobre el GC
  • Mala localización en memoria (cache-unfriendly)


Incluso algo simple como una lista de enteros:


List<Integer>


Es en realidad:

List -> referencias -> objetos Integer


¿Qué propone Valhalla?


Valhalla introduce un nuevo tipo de clases: Clases inline (value classes)


Ejemplo conceptual:


value class Point {

    int x;

    int y;

}


Esto permite:

  • Sin identidad (no == por referencia)
  • Sin overhead de objeto
  • Layout compacto en memoria


¿Por qué esto es importante?


Mejora de rendimiento

  • Menos uso de heap
  • Mejor uso de CPU cache
  • Menos GC


Estructuras más eficientes


Ejemplo:

List<Point>


Con Valhalla podría ser:

[ x1, y1, x2, y2, x3, y3 ]


👉 En lugar de:

[ ref -> Point, ref -> Point, ref -> Point ]


Conceptos clave de Valhalla


Value Objects

  • No tienen identidad, la identidad esta dada por su propio valor
  • Son inmutables (conceptualmente)
  • Se comparan por contenido


Flattening

El JVM puede “aplanar” estructuras:


class Line {

    Point p1;

    Point p2;

}


En memoria:

[x1, y1, x2, y2]


Specialized Generics (futuro)

Uno de los grandes problemas actuales:

List<int> ❌

List<Integer> ✅ (pero lento)


Valhalla busca permitir:

List<int> ✅


Sin boxing


Cambios importantes a tener en cuenta

  • No todo será automáticamente value type
  • Cambia el modelo mental (menos identidad, más datos)
  • Impacta librerías y frameworks


Project Valhalla aún está en desarrollo activo.

No es parte estable de Java todavía, pero ya hay:

  • Prototipos en OpenJDK
  • Features en incubación


Veamos un ejemplo: 


 Antes (Java actual)


class Complex {

    double re;

    double im;

}


  • Objeto en heap
  • Referencias


Con Valhalla (ideal)


value class Complex {

    double re;

    double im;

}


  • Datos contiguos
  • Sin overhead




Project Valhalla busca algo muy ambicioso:

Combinar lo mejor de OOP con la eficiencia de lenguajes de bajo nivel


En otras palabras:

  • Mantener la simplicidad de Java
  • Reducir el costo de abstracción


En C# tenemos el struct que es muy similar a lo que busca Valhalla.