8 min read

Cómo optimizar tu base de datos: 8 técnicas efectivas

Table of Contents

¿Estás lidiando con desafíos de rendimiento en tus sistemas de software?

A menudo, el problema es en la base de datos. Optimizar esta pieza clave es vital para alcanzar un óptimo rendimiento de tus sistemas. Para ayudarte, aquí te presento 8 técnicas avanzadas y efectivas para optimizar bases de datos, garantizando así sistemas más rápidos y eficientes.

Normalización y desnormalización

Normalización: Este método estructura los datos de manera que se reduce la repetición y la interdependencia, disminuyendo así el riesgo de inconsistencias en los datos. Al dividir tablas grandes en otras más pequeñas y conectadas entre sí, la normalización promueve la uniformidad de los datos. No obstante, este proceso puede resultar en la necesidad de realizar consultas más elaboradas.

CREATE TABLE vehicles (
    vehicle_id INT PRIMARY KEY,
    make VARCHAR(50),
    model VARCHAR(50),
    year INT
);

CREATE TABLE maintenance_records (
    record_id INT PRIMARY KEY,
    vehicle_id INT,
    service_date DATE,
    description VARCHAR(255),
    FOREIGN KEY (vehicle_id) REFERENCES vehicles(vehicle_id)
);

Desnormalización: A diferencia de la normalización, que busca minimizar la repetición de datos, la desnormalización implementa deliberadamente cierta redundancia para optimizar la eficiencia de las consultas, en particular en tareas con un alto volumen de lecturas. Este proceso puede incluir la incorporación de columnas o tablas redundantes de manera planificada.

CREATE TABLE denormalized_maintenance (
    record_id INT PRIMARY KEY,
    vehicle_id INT,
    make VARCHAR(50),
    model VARCHAR(50),
    year INT,
    service_date DATE,
    description VARCHAR(255)
);

Indexacion

Establecer índices: Los índices son sistemas de organización de datos que facilitan la búsqueda eficiente, lo que resulta en una mejora notable en la velocidad de las consultas. Operan mediante la creación de una estructura de datos ordenada, que permite al sistema de gestión de la base de datos encontrar de manera rápida las filas que cumplen con una condición específica en una cláusula WHERE. Aunque estos índices incrementan la velocidad de las consultas SELECT, pueden disminuir la eficiencia en operaciones de escritura. Por tanto, es esencial mantener un equilibrio adecuado entre el rendimiento de las operaciones de lectura y escritura.

CREATE INDEX idx_vehicle_license ON vehicles(license_plate);

Implementar índices combinados: Los índices combinados se construyen utilizando varias columnas, lo que resulta beneficioso para ejecutar consultas que requieren filtrar o ordenar utilizando varias condiciones simultáneamente. Esta estrategia minimiza la necesidad de tener índices individuales para cada columna, optimizando así el desempeño del planificador de consultas.

CREATE INDEX idx_brand_model ON vehicles(brand, model);

Segmentación de tablas

Dividir tablas en particiones: La estrategia de segmentar tablas extensas en segmentos más reducidos y fáciles de gestionar puede potenciar notablemente la eficiencia de las consultas. Al hacer esto, se permite que el sistema de la base de datos opere sobre porciones de datos más limitadas, agilizando así la ejecución de las consultas.

CREATE TABLE maintenance_records (
    record_id INT PRIMARY KEY,
    vehicle_id INT,
    service_date DATE,
    description VARCHAR(255)
) PARTITION BY RANGE (service_date);

CREATE TABLE maintenance_records_y1990 PARTITION OF maintenance_records
    FOR VALUES FROM ('1980-01-01') TO ('1990-01-01');

CREATE TABLE maintenance_records_y2000 PARTITION OF maintenance_records
    FOR VALUES FROM ('1990-01-01') TO ('2000-01-01');

CREATE TABLE maintenance_records_y2010 PARTITION OF maintenance_records
    FOR VALUES FROM ('2000-01-01') TO ('2010-01-01');

CREATE TABLE maintenance_records_y2020 PARTITION OF maintenance_records
    FOR VALUES FROM ('2010-01-01') TO ('2020-01-01');

CREATE TABLE maintenance_records_y2021 PARTITION OF maintenance_records
    FOR VALUES FROM ('2020-01-01') TO (MAXVALUE);

Eliminación de particiones irrelevantes: Es importante que el SELECT descarte las particiones que no son relevantes durante la ejecución de una consulta. Al hacer esto, se evita la necesidad de revisar todos los datos, lo que resulta en un aumento de la eficiencia y rapidez de la consulta.

SELECT * FROM maintenance_records WHERE service_date >= '2022-01-01' AND service_date < '2023-01-01';

Mejora de consultas

Perfeccionar consultas: Es crucial realizar un análisis y optimización constante de las consultas que se ejecutan con frecuencia. Herramientas como EXPLAIN son útiles para desglosar y entender cómo se ejecutan las consultas, lo que permite detectar y mejorar aspectos que no son eficientes.

EXPLAIN SELECT * FROM maintenance_records WHERE vehicle_id = 987;

Restringir el Uso de SELECT *: Es preferible especificar y recuperar solo las columnas que son realmente necesarias, en lugar de utilizar un SELECT * para obtener todas. Al hacer esto, se minimiza la transferencia y el procesamiento de datos innecesarios, lo cual contribuye a la eficiencia de la consulta.

EXPLAIN SELECT plate, type FROM maintenance_records WHERE vehicle_id = 987;

Caching

Almacenamiento en caché de Consultas: Establece un sistema de caché que guarde los resultados de las consultas que se realizan con regularidad. Al hacer esto, se disminuye el trabajo de la base de datos, ya que se pueden entregar resultados previamente guardados, lo que a su vez acelera los tiempos de respuesta.

-- Pseudocódigo para la integración de PostgreSQL con Redis

-- Definir la clave de caché y la variable para almacenar el resultado
DECLARE cacheKey TEXT := 'query_cache_vehicles';
DECLARE cachedResult RECORD;

-- Intentar obtener el resultado de Redis
cachedResult := REDIS_GET(cacheKey);

-- Comprobar si el resultado está en caché
IF cachedResult IS NULL THEN
    -- Si no está en caché, ejecutar la consulta en PostgreSQL
    cachedResult := EXECUTE_QUERY('SELECT * FROM vehicles WHERE year > 2010');

    -- Almacenar el resultado en Redis con un tiempo de expiración
    REDIS_SET(cacheKey, cachedResult, EXPIRY_TIME);
END IF;

-- Usar cachedResult para el procesamiento posterior

Almacenamiento en caché de objetos: Guarda en una caché aquellos objetos o información que se consulta con regularidad dentro de la capa de aplicación. Esto ayuda a reducir el número de consultas a la base de datos. Para ello, se pueden utilizar herramientas especializadas de caché en memoria o frameworks diseñados para este propósito.

import { get, set } from 'redis'; // Suponiendo que estamos usando un cliente Redis como 'redis'

// Esta función asume que tienes una función para obtener datos del usuario de tu base de datos
async function fetchUserDataFromDatabase(userId: string): Promise<UserData> {
    // Aquí iría la lógica para buscar los datos del usuario en la base de datos
    // Por ejemplo, podría ser una consulta a una base de datos PostgreSQL
}

async function getUserData(userId: string): Promise<UserData> {
    // Intentar obtener los datos del usuario del caché
    const cacheKey = `user_${userId}`;
    let userData = await get(cacheKey);

    if (userData === null) {
        // Si los datos no están en caché, obtenerlos de la base de datos
        userData = await fetchUserDataFromDatabase(userId);

        // Almacenar los datos en el caché para futuras solicitudes
        await set(cacheKey, JSON.stringify(userData), 'EX', TIMEOUT);
    } else {
        // Convertir los datos de usuario de la cadena JSON al objeto
        userData = JSON.parse(userData);
    }

    return userData;
}
💡

Esta opción resulta ser más eficiente cuando la lógica de la base de datos se mantiene separada y las validaciones del caché se realizan directamente en el servidor. Esta separación de responsabilidades no solo optimiza el rendimiento, sino que también facilita la mantenibilidad y escalabilidad del sistema.

Mantenimiento regular

Renovación de Estadísticas: Es esencial mantener al día las estadísticas para que el planificador de consultas pueda desarrollar planes de ejecución basados en información precisa. Al actualizar estas estadísticas con frecuencia, se asegura una optimización de consultas más exacta y eficaz.

ANALYZE vehicles;

Gestión de datos históricos: Transfiere o depura datos obsoletos que ya no son de utilidad. Esta práctica puede agilizar la ejecución de consultas y disminuir la demanda de espacio de almacenamiento, lo cual es particularmente beneficioso en sistemas que manejan amplias bases de datos históricas.

Optimización del hardware

Afinar la Configuración del Servidor de Base de Datos: Modifica la configuración del servidor de base de datos para adaptarse mejor a la carga de trabajo actual y las capacidades de tu equipo. Esto abarca ajustes como el tamaño de los buffers, la configuración de la caché y los límites de las conexiones.

- Ejemplo: Expandir la capacidad de la caché de consultas
SET GLOBAL query_cache_size = 256M;

Implementar SSDs: Evalúa la posibilidad de usar Unidades de Estado Sólido (SSDs) para el almacenamiento. Los SSDs, al proporcionar un acceso más ágil a los datos que los Discos Duros tradicionales (HDDs), pueden incrementar significativamente la eficiencia de la base de datos.

Connection pooling

Implementar el pool de conexiones: Emplea el pool de conexiones para reutilizar las conexiones existentes a la base de datos en lugar de crear nuevas para cada petición. Este método es eficaz para administrar las conexiones a la base de datos, optimizando su uso y reduciendo la carga adicional que supone establecer conexiones nuevas constantemente.

import { createConnection } from "typeorm";

createConnection({
  type: "mysql",
  host: "localhost",
  port: 3306,
  username: "username",
  password: "password",
  database: "database",
  entities: [
    // Aquí irían tus entidades
  ],
  synchronize: true,
  logging: false,
  // Configuración del pool de conexiones
  extra: {
    poolSize: 10, // El número máximo de conexiones en el pool
  },
})
  .then((connection) => {
    // Aquí puedes empezar a trabajar con tu conexión
    console.log("Conexión a la base de datos establecida");
  })
  .catch((error) =>
    console.log("Error al conectar a la base de datos:", error),
  );

Database Design

Desarrollo de un esquema de Base de datos orientado al Rendimiento: Al estructurar el esquema de tu base de datos, prioriza la eficiencia. Esto implica seleccionar cuidadosamente los tipos de datos, aplicar restricciones adecuadas y evitar relaciones superfluas. Un esquema diseñado estratégicamente puede mejorar notablemente la eficacia en la ejecución de consultas.

CREATE TABLE vehicles (
    vehicle_id INT PRIMARY KEY,
    make VARCHAR(50),
    model VARCHAR(50),
    year INT,
    color VARCHAR(30),
    price DECIMAL(10, 2)
    -- Se pueden añadir columnas adicionales según sea necesario
);

Existen múltiples estrategias para optimizar una base de datos, siendo crucial mantener un control riguroso y una monitorización efectiva. Además, es esencial incorporar procedimientos de mantenimiento regular y actualizaciones. La aplicación de estas técnicas debe adaptarse al contexto específico y a las necesidades únicas de tu proyecto. Por ejemplo, la implementación de índices eficientes, la normalización de datos para evitar redundancias, o incluso la desnormalización en casos donde el rendimiento de lectura es prioritario, son algunas de las muchas tácticas a considerar. ¿Conoces otras formas de optimización de bases de datos?

Estoy interesado en escuchar tus ideas.

Gracias por leer