0. Conceptos básicos de LLM
Preentrenamiento
El preentrenamiento es la fase fundamental en el desarrollo de un modelo de lenguaje grande (LLM) donde el modelo se expone a grandes y diversas cantidades de datos textuales. Durante esta etapa, el LLM aprende las estructuras, patrones y matices fundamentales del lenguaje, incluyendo gramática, vocabulario, sintaxis y relaciones contextuales. Al procesar estos datos extensos, el modelo adquiere una comprensión amplia del lenguaje y del conocimiento general del mundo. Esta base integral permite al LLM generar texto coherente y contextualmente relevante. Posteriormente, este modelo preentrenado puede someterse a un ajuste fino, donde se entrena aún más en conjuntos de datos especializados para adaptar sus capacidades a tareas o dominios específicos, mejorando su rendimiento y relevancia en aplicaciones específicas.
Componentes principales de LLM
Usualmente, un LLM se caracteriza por la configuración utilizada para entrenarlo. Estos son los componentes comunes al entrenar un LLM:
- Parámetros: Los parámetros son los pesos y sesgos aprendibles en la red neuronal. Estos son los números que el proceso de entrenamiento ajusta para minimizar la función de pérdida y mejorar el rendimiento del modelo en la tarea. Los LLMs suelen utilizar millones de parámetros.
- Longitud del contexto: Esta es la longitud máxima de cada oración utilizada para preentrenar el LLM.
- Dimensión de incrustación: El tamaño del vector utilizado para representar cada token o palabra. Los LLMs suelen usar miles de millones de dimensiones.
- Dimensión oculta: El tamaño de las capas ocultas en la red neuronal.
- Número de capas (profundidad): Cuántas capas tiene el modelo. Los LLMs suelen usar decenas de capas.
- Número de cabezas de atención: En los modelos de transformador, esta es la cantidad de mecanismos de atención separados que se utilizan en cada capa. Los LLMs suelen usar decenas de cabezas.
- Dropout: El dropout es algo así como el porcentaje de datos que se eliminan (las probabilidades se convierten en 0) durante el entrenamiento utilizado para prevenir el sobreajuste. Los LLMs suelen usar entre 0-20%.
Configuración del modelo GPT-2:
GPT_CONFIG_124M = {
"vocab_size": 50257, // Vocabulary size of the BPE tokenizer
"context_length": 1024, // Context length
"emb_dim": 768, // Embedding dimension
"n_heads": 12, // Number of attention heads
"n_layers": 12, // Number of layers
"drop_rate": 0.1, // Dropout rate: 10%
"qkv_bias": False // Query-Key-Value bias
}
Tensors en PyTorch
En PyTorch, un tensor es una estructura de datos fundamental que sirve como un array multidimensional, generalizando conceptos como escalares, vectores y matrices a dimensiones potencialmente más altas. Los tensores son la forma principal en que los datos se representan y manipulan en PyTorch, especialmente en el contexto del aprendizaje profundo y las redes neuronales.
Concepto Matemático de Tensores
- Escalares: Tensores de rango 0, que representan un solo número (cero-dimensional). Como: 5
- Vectores: Tensores de rango 1, que representan un array unidimensional de números. Como: [5,1]
- Matrices: Tensores de rango 2, que representan arrays bidimensionales con filas y columnas. Como: [[1,3], [5,2]]
- Tensores de Rango Superior: Tensores de rango 3 o más, que representan datos en dimensiones superiores (por ejemplo, tensores 3D para imágenes en color).
Tensores como Contenedores de Datos
Desde una perspectiva computacional, los tensores actúan como contenedores para datos multidimensionales, donde cada dimensión puede representar diferentes características o aspectos de los datos. Esto hace que los tensores sean altamente adecuados para manejar conjuntos de datos complejos en tareas de aprendizaje automático.
Tensores de PyTorch vs. Arrays de NumPy
Mientras que los tensores de PyTorch son similares a los arrays de NumPy en su capacidad para almacenar y manipular datos numéricos, ofrecen funcionalidades adicionales cruciales para el aprendizaje profundo:
- Diferenciación Automática: Los tensores de PyTorch soportan el cálculo automático de gradientes (autograd), lo que simplifica el proceso de cálculo de derivadas requeridas para entrenar redes neuronales.
- Aceleración por GPU: Los tensores en PyTorch pueden ser movidos y computados en GPUs, acelerando significativamente los cálculos a gran escala.
Creando Tensores en PyTorch
Puedes crear tensores usando la función torch.tensor
:
pythonCopy codeimport torch
# Scalar (0D tensor)
tensor0d = torch.tensor(1)
# Vector (1D tensor)
tensor1d = torch.tensor([1, 2, 3])
# Matrix (2D tensor)
tensor2d = torch.tensor([[1, 2],
[3, 4]])
# 3D Tensor
tensor3d = torch.tensor([[[1, 2], [3, 4]],
[[5, 6], [7, 8]]])
Tipos de Datos de Tensor
Los tensores de PyTorch pueden almacenar datos de varios tipos, como enteros y números de punto flotante.
Puedes verificar el tipo de dato de un tensor usando el atributo .dtype
:
tensor1d = torch.tensor([1, 2, 3])
print(tensor1d.dtype) # Output: torch.int64
- Los tensores creados a partir de enteros de Python son de tipo
torch.int64
. - Los tensores creados a partir de flotantes de Python son de tipo
torch.float32
.
Para cambiar el tipo de datos de un tensor, utiliza el método .to()
:
float_tensor = tensor1d.to(torch.float32)
print(float_tensor.dtype) # Output: torch.float32
Operaciones Comunes con Tensores
PyTorch proporciona una variedad de operaciones para manipular tensores:
- Accediendo a la Forma: Usa
.shape
para obtener las dimensiones de un tensor.
print(tensor2d.shape) # Salida: torch.Size([2, 2])
- Reformateando Tensores: Usa
.reshape()
o.view()
para cambiar la forma.
reshaped = tensor2d.reshape(4, 1)
- Transponiendo Tensores: Usa
.T
para transponer un tensor 2D.
transposed = tensor2d.T
- Multiplicación de Matrices: Usa
.matmul()
o el operador@
.
result = tensor2d @ tensor2d.T
Importancia en el Aprendizaje Profundo
Los tensores son esenciales en PyTorch para construir y entrenar redes neuronales:
- Almacenan datos de entrada, pesos y sesgos.
- Facilitan las operaciones requeridas para los pasos hacia adelante y hacia atrás en los algoritmos de entrenamiento.
- Con autograd, los tensores permiten el cálculo automático de gradientes, agilizando el proceso de optimización.
Diferenciación Automática
La diferenciación automática (AD) es una técnica computacional utilizada para evaluar las derivadas (gradientes) de funciones de manera eficiente y precisa. En el contexto de redes neuronales, AD permite el cálculo de gradientes requeridos para algoritmos de optimización como el descenso de gradiente. PyTorch proporciona un motor de diferenciación automática llamado autograd que simplifica este proceso.
Explicación Matemática de la Diferenciación Automática
1. La Regla de la Cadena
En el corazón de la diferenciación automática está la regla de la cadena del cálculo. La regla de la cadena establece que si tienes una composición de funciones, la derivada de la función compuesta es el producto de las derivadas de las funciones compuestas.
Matemáticamente, si y=f(u)
y u=g(x)
, entonces la derivada de y
con respecto a x
es:
2. Grafo Computacional
En AD, los cálculos se representan como nodos en un grafo computacional, donde cada nodo corresponde a una operación o una variable. Al recorrer este grafo, podemos calcular derivadas de manera eficiente.
- Ejemplo
Consideremos una función simple:
Donde:
σ(z)
es la función sigmoide.y=1.0
es la etiqueta objetivo.L
es la pérdida.
Queremos calcular el gradiente de la pérdida L
con respecto al peso w
y al sesgo b
.
4. Cálculo de Gradientes Manualmente
5. Cálculo Numérico
Implementando la Diferenciación Automática en PyTorch
Ahora, veamos cómo PyTorch automatiza este proceso.
pythonCopy codeimport torch
import torch.nn.functional as F
# Define input and target
x = torch.tensor([1.1])
y = torch.tensor([1.0])
# Initialize weights with requires_grad=True to track computations
w = torch.tensor([2.2], requires_grad=True)
b = torch.tensor([0.0], requires_grad=True)
# Forward pass
z = x * w + b
a = torch.sigmoid(z)
loss = F.binary_cross_entropy(a, y)
# Backward pass
loss.backward()
# Gradients
print("Gradient w.r.t w:", w.grad)
print("Gradient w.r.t b:", b.grad)
Salida:
cssCopy codeGradient w.r.t w: tensor([-0.0898])
Gradient w.r.t b: tensor([-0.0817])
Retropropagación en Redes Neuronales Más Grandes
1. Ampliación a Redes Multicapa
En redes neuronales más grandes con múltiples capas, el proceso de cálculo de gradientes se vuelve más complejo debido al aumento en el número de parámetros y operaciones. Sin embargo, los principios fundamentales permanecen iguales:
- Paso Adelante: Calcular la salida de la red pasando las entradas a través de cada capa.
- Calcular Pérdida: Evaluar la función de pérdida utilizando la salida de la red y las etiquetas objetivo.
- Paso Atrás (Retropropagación): Calcular los gradientes de la pérdida con respecto a cada parámetro en la red aplicando la regla de la cadena de manera recursiva desde la capa de salida hasta la capa de entrada.
2. Algoritmo de Retropropagación
- Paso 1: Inicializar los parámetros de la red (pesos y sesgos).
- Paso 2: Para cada ejemplo de entrenamiento, realizar un paso adelante para calcular las salidas.
- Paso 3: Calcular la pérdida.
- Paso 4: Calcular los gradientes de la pérdida con respecto a cada parámetro utilizando la regla de la cadena.
- Paso 5: Actualizar los parámetros utilizando un algoritmo de optimización (por ejemplo, descenso de gradiente).
3. Representación Matemática
Considera una red neuronal simple con una capa oculta:
4. Implementación en PyTorch
PyTorch simplifica este proceso con su motor autograd.
import torch
import torch.nn as nn
import torch.optim as optim
# Define a simple neural network
class SimpleNet(nn.Module):
def __init__(self):
super(SimpleNet, self).__init__()
self.fc1 = nn.Linear(10, 5) # Input layer to hidden layer
self.relu = nn.ReLU()
self.fc2 = nn.Linear(5, 1) # Hidden layer to output layer
self.sigmoid = nn.Sigmoid()
def forward(self, x):
h = self.relu(self.fc1(x))
y_hat = self.sigmoid(self.fc2(h))
return y_hat
# Instantiate the network
net = SimpleNet()
# Define loss function and optimizer
criterion = nn.BCELoss()
optimizer = optim.SGD(net.parameters(), lr=0.01)
# Sample data
inputs = torch.randn(1, 10)
labels = torch.tensor([1.0])
# Training loop
optimizer.zero_grad() # Clear gradients
outputs = net(inputs) # Forward pass
loss = criterion(outputs, labels) # Compute loss
loss.backward() # Backward pass (compute gradients)
optimizer.step() # Update parameters
# Accessing gradients
for name, param in net.named_parameters():
if param.requires_grad:
print(f"Gradient of {name}: {param.grad}")
En este código:
- Forward Pass: Calcula las salidas de la red.
- Backward Pass:
loss.backward()
calcula los gradientes de la pérdida con respecto a todos los parámetros. - Parameter Update:
optimizer.step()
actualiza los parámetros en función de los gradientes calculados.
5. Comprendiendo el Backward Pass
Durante el backward pass:
- PyTorch recorre el grafo computacional en orden inverso.
- Para cada operación, aplica la regla de la cadena para calcular los gradientes.
- Los gradientes se acumulan en el atributo
.grad
de cada tensor de parámetro.
6. Ventajas de la Diferenciación Automática
- Eficiencia: Evita cálculos redundantes al reutilizar resultados intermedios.
- Precisión: Proporciona derivadas exactas hasta la precisión de la máquina.
- Facilidad de Uso: Elimina el cálculo manual de derivadas.