C

Tutorial C: Estructuras en C

C: Aprende a definir y usar estructuras para organizar y acceder a datos de forma eficiente.

Aprende C GRATIS y certifícate

Declaración y definición de estructuras

Las estructuras permiten agrupar múltiples variables de distintos tipos bajo un mismo identificador, fomentando un diseño más organizado del código. Con la palabra clave struct, se pueden crear tipos compuestos que representan entidades lógicas como un registro, un vector de atributos o cualquier objeto con propiedades heterogéneas.

Para definir una estructura, se utiliza la siguiente sintaxis, donde primero se especifica la palabra clave struct, seguida de un nombre opcional y, dentro de llaves, la lista de miembros con sus tipos y nombres:

struct Persona {
    char nombre[50];
    int edad;
    float altura;
};

También es posible omitir el nombre de la estructura, aunque en desarrollos complejos se recomienda mantenerlo para reutilizar ese tipo en distintos puntos del código.

Separar la declaración de la definición resulta útil en proyectos grandes: la declaración se limita a exponer la existencia del tipo (por ejemplo, struct Persona; sin miembros) para referencias adelantadas, mientras que la definición cubre detalladamente cada campo que lo compone. Esto ayuda a mantener un orden claro en los archivos de cabecera y en las implementaciones.

En entornos donde se busca modularidad, se suele colocar la definición de la estructura en un único archivo de cabecera para que el resto de módulos puedan incluirla sin redundancias. Así, se evitan copias innecesarias del mismo tipo y se facilita la gestión de cambios posteriores en los miembros de la estructura.

Acceso a miembros de la estructura

Para acceder a los campos de una estructura cuando se trabaja con una variable directa, se emplea el operador .. Este operador indica que se está usando el valor de la variable, sin necesidad de ningún tipo de desreferencia. Por ejemplo, si se dispone de un objeto persona1 y se desea asignar la edad y altura, basta con realizar:

persona1.edad = 30;
persona1.altura = 1.75f;

En este caso, el operador . proporciona un acceso directo a los miembros internos, ya que la variable persona1 no es un puntero.

La situación cambia cuando se manipula un puntero a una estructura. En ese contexto, el operador -> facilita la desreferencia interna hacia el miembro correspondiente. La flecha combina el acceso indirecto con la selección del campo, lo que evita escribir (*puntero).miembro. Si se tiene struct Persona *pPersona = &persona1;, se podría modificar un atributo así:

pPersona->edad = 35;
pPersona->altura = 1.80f;

De este modo, -> realiza internamente la desreferencia necesaria y permite tratar a cada miembro como si fuera parte de la estructura apuntada.

Cuando se accede a campos dentro de un programa complejo, es recomendable mantener una consistencia de estilo para que sea más legible el uso de . y ->. Desplegar cada asignación o lectura en una línea separada ayuda a identificar posibles errores y a comprender con claridad el flujo de acceso a los datos. Lista de sugerencias:

  • Mantener una nomenclatura clara en las variables y punteros para diferenciar fácilmente cada tipo de acceso.
  • Utilizar -> únicamente si se trabaja con un puntero o referencia hacia la estructura, evitando confusiones en el código.
  • Acompañar cada acceso a miembros con comentarios breves pero significativos, sobre todo cuando se modifiquen varios campos consecutivos.

Estructuras anidadas

Las estructuras anidadas permiten representar relaciones internas más específicas al encapsular una estructura dentro de otra. Este enfoque es útil cuando se requiere organizar datos de forma jerárquica o describir atributos que pertenecen a distintas categorías sin recurrir a múltiples tipos externos.

Un ejemplo ilustrativo consiste en agrupar la información de una dirección dentro de la estructura de una persona. Así, se pueden definir los campos de la siguiente manera:

struct Direccion {
    char calle[50];
    int numero;
};

struct Persona {
    char nombre[50];
    struct Direccion direccion;
};

En el caso anterior, se accede a cada subcampo utilizando el operador . según corresponda. Si existe una variable persona1 de tipo struct Persona, para asignar el nombre de la calle se puede emplear:

strcpy(persona1.direccion.calle, "Avenida Central");
persona1.direccion.numero = 123;

Cada miembro de la subestructura queda así integrado de manera coherente bajo el ámbito de Persona.

Al diseñar estructuras anidadas, conviene mantener una distribución coherente de los campos, de modo que las responsabilidades de cada parte estén bien delimitadas. Algunos consejos valiosos incluyen:

  • Diseñar la lógica de cada estructura en torno a un objetivo claro.
  • Agrupar los campos relacionados en subestructuras para facilitar la lectura y la reutilización de datos.
  • Revisar la distribución de tamaños y tipos para asegurar un control óptimo de la memoria.

Arreglos y punteros a estructuras

Cuando se utilizan arreglos de estructuras, todos los elementos comparten el mismo tipo de struct, pero cada posición mantiene valores independientes en sus campos. Por ejemplo, se puede declarar un array local de struct Persona para gestionar un grupo de hasta 100 registros, cada uno con sus propias características. Una aproximación común es recorrer el array con un bucle for, asignar valores a cada miembro y, si es necesario, aplicar operaciones de búsqueda o filtrado sobre cada estructura.

Un arreglo de estructuras también puede combinarse con punteros a estructuras para acceder a regiones concretas del array o para pasarlo como argumento a funciones. Por norma general, se podría utilizar un puntero para apuntar al primer elemento del array y así iterar sobre él sin necesidad de índices. Esto permite escribir expresiones más compactas usando el operador ->. Un esquema típico de acceso es:

#include <stdio.h>

struct Persona
{
    int edad;
    float altura;
};

int main(void)
{
    struct Persona grupo[5];
    struct Persona *p = grupo; // p apunta al inicio del array

    for (int i = 0; i < 5; i++)
    {
        p->edad = 20 + i;
        p->altura = 1.70f + 0.01f * i;
        p++;
    }
    for (int i = 0; i < 5; i++)
    {
        printf("Persona %d: Edad: %d, Altura: %f\n", i, grupo[i].edad, grupo[i].altura);
    }

    return 0;
}

En este bloque, el puntero avanza a través de cada elemento y desreferencia los campos con el operador ->, sin necesidad de utilizar grupo[i].edad o grupo[i].altura.

Cuando se pasan arreglos de estructuras a funciones, es habitual usar la forma funcion(struct Persona *arr, int tam), de manera que dentro de la función se opere con el puntero a la primera posición y un entero que indica el tamaño del array. Esta estrategia resulta útil para generar rutinas de inicialización, clasificación o búsqueda de datos sin duplicar la memoria. Además, al manejar punteros, se tiene la flexibilidad de aplicar la misma función a segmentos concretos del array, restringiendo el ámbito según sea necesario.

En escenarios donde se requiere memoria dinámica, se puede reservar espacio para un conjunto de estructuras con malloc() y tratar la dirección de retorno como un puntero al primer elemento del array. Por ejemplo:

#include <stdio.h>
#include <stdlib.h>

struct Persona
{
    int edad;
    float altura;
};

int main(void)
{
    struct Persona *conjunto = malloc(10 * sizeof(struct Persona));
    if (conjunto != NULL)
    {
        // Inicializar cada estructura...
        // Uso de conjunto[i] o (conjunto + i)->campo
        for (int i = 0; i < 10; i++)
        {
            conjunto[i].edad = i;
            (conjunto + i)->altura = 1.0f + i;
        }
    }

    for (int i = 0; i < 10; i++)
    {
        printf("Edad: %d, Altura: %f\n", (conjunto + i)->edad, conjunto[i].altura);
    }

    free(conjunto);

    return 0;
}

Con este patrón, la gestión de la memoria asignada queda bajo control del desarrollador, resultando práctico para trabajos en los que el tamaño del conjunto de datos no se conoce en tiempo de compilación.

Aprende C GRATIS online

Todas las lecciones de C

Accede a todas las lecciones de C y aprende con ejemplos prácticos de código y ejercicios de programación con IDE web sin instalar nada.

Introducción A Css

CSS3

Introducción Y Entorno

Sintaxis

CSS3

Sintaxis De Selectores Y Propiedades

Selectores Básicos

CSS3

Sintaxis De Selectores Y Propiedades

Herencia Y Cascada

CSS3

Sintaxis De Selectores Y Propiedades

Pseudo-clases Y Pseudo-elementos

CSS3

Sintaxis De Selectores Y Propiedades

Estilos De Fuente

CSS3

Estilización De Texto Y Fondo

Propiedades De Texto

CSS3

Estilización De Texto Y Fondo

Sombras En Texto Y Cajas

CSS3

Estilización De Texto Y Fondo

Propiedades De Fondo

CSS3

Estilización De Texto Y Fondo

Modelo De Caja

CSS3

Modelo Caja Y Posicionamiento

Propiedades De Posicionamiento

CSS3

Modelo Caja Y Posicionamiento

Propiedad 'Display'

CSS3

Modelo Caja Y Posicionamiento

Elementos 'Float' Y 'Clear'

CSS3

Modelo Caja Y Posicionamiento

Flexbox Para Crear Layouts Y Estructuras

CSS3

Flexbox Y Grid

Css Grid Para Crear Layouts Y Estructuras

CSS3

Flexbox Y Grid

Animaciones Y Transiciones

CSS3

Técnicas Modernas Y Metodologías

Variables En Css

CSS3

Técnicas Modernas Y Metodologías

Diseño Responsive Con Media Queries

CSS3

Técnicas Modernas Y Metodologías

Metodologías De Escritura En Css

CSS3

Técnicas Modernas Y Metodologías

Introducción A Javascript

JavaScript

Sintaxis

Tipos De Datos

JavaScript

Sintaxis

Variables

JavaScript

Sintaxis

Operadores

JavaScript

Sintaxis

Estructuras De Control

JavaScript

Sintaxis

Funciones

JavaScript

Sintaxis

Funciones Cierre (Closure)

JavaScript

Sintaxis

Funciones Flecha

JavaScript

Programación Funcional

Filtrado Con Filter() Y Find()

JavaScript

Programación Funcional

Transformación Con Map()

JavaScript

Programación Funcional

Reducción Con Reduce()

JavaScript

Programación Funcional

Clases Y Objetos

JavaScript

Programación Orientada A Objetos

Excepciones

JavaScript

Programación Orientada A Objetos

Encapsulación

JavaScript

Programación Orientada A Objetos

Herencia

JavaScript

Programación Orientada A Objetos

Polimorfismo

JavaScript

Programación Orientada A Objetos

Array

JavaScript

Estructuras De Datos

Conjuntos Con Set

JavaScript

Estructuras De Datos

Mapas Con Map

JavaScript

Estructuras De Datos

Manipulación Dom

JavaScript

Dom

Selección De Elementos Dom

JavaScript

Dom

Modificación De Elementos Dom

JavaScript

Dom

Eventos Del Dom

JavaScript

Dom

Callbacks

JavaScript

Programación Asíncrona

Promises

JavaScript

Programación Asíncrona

Async / Await

JavaScript

Programación Asíncrona

Introducción A Typescript

TypeScript

Introducción Y Entorno

Variables Y Constantes

TypeScript

Sintaxis

Operadores

TypeScript

Sintaxis

Control De Flujo

TypeScript

Sintaxis

Funciones

TypeScript

Sintaxis

Funciones Flecha

TypeScript

Sintaxis

Clases Y Objetos

TypeScript

Programación Orientada A Objetos

Interfaces

TypeScript

Programación Orientada A Objetos

Encapsulación

TypeScript

Programación Orientada A Objetos

Herencia

TypeScript

Programación Orientada A Objetos

Polimorfismo

TypeScript

Programación Orientada A Objetos

Inmutabilidad

TypeScript

Programación Funcional

Funciones Puras

TypeScript

Programación Funcional

Funciones De Primera Clase

TypeScript

Programación Funcional

Funciones De Alto Orden

TypeScript

Programación Funcional

Tipos Literales

TypeScript

Tipos De Datos Avanzados

Tipos Genéricos

TypeScript

Tipos De Datos Avanzados

Tipos De Unión E Intersección

TypeScript

Tipos De Datos Avanzados

Tipos De Utilidad

TypeScript

Tipos De Datos Avanzados

Módulos

TypeScript

Namespaces Y Módulos

Namespaces

TypeScript

Namespaces Y Módulos

Resolución De Módulos

TypeScript

Namespaces Y Módulos

Introducción A C#

Sintaxis

Creación De Proyecto C#

Sintaxis

Variables Y Constantes

Sintaxis

Tipos De Datos

Sintaxis

Operadores

Sintaxis

Control De Flujo

Sintaxis

Funciones

Sintaxis

Clases Y Encapsulación

Programación Orientada A Objetos

Objetos

Programación Orientada A Objetos

Constructores Y Destructores

Programación Orientada A Objetos

Herencia

Programación Orientada A Objetos

Polimorfismo

Programación Orientada A Objetos

Excepciones

Excepciones

Arrays Y Listas

Colecciones Y Linq

Diccionarios

Colecciones Y Linq

Conjuntos, Colas Y Pilas

Colecciones Y Linq

Uso De Consultas Linq

Colecciones Y Linq

Delegados

Programación Asíncrona

Eventos

Programación Asíncrona

Lambdas

Programación Asíncrona

Uso De Async Y Await

Programación Asíncrona

Tareas

Programación Asíncrona

Introducción A C

Introducción Y Entorno

Primer Programa En C

Introducción Y Entorno

Estructura Básica De Un Programa En C

Sintaxis

Operadores Y Expresiones

Sintaxis

Control De Flujo

Sintaxis

Arrays Y Manejo De Cadenas

Sintaxis

Funciones

Funciones Y Punteros

Punteros

Funciones Y Punteros

Gestión De Memoria Dinámica

Funciones Y Punteros

Estructuras En C

Estructuras, Uniones Y Tipos

Uniones Y Enumeraciones

Estructuras, Uniones Y Tipos

Introducción A Tailwind Css

Tailwind CSS

Introducción Y Entorno

Instalación De Tailwind Css

Tailwind CSS

Introducción Y Entorno

Fundamentos Del Sistema De Utility-first

Tailwind CSS

Fundamentos

Fundamentos Del Diseño Responsive

Tailwind CSS

Fundamentos

Tipografía Y Fuentes En Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Tamaño De Tailwind Css

Tailwind CSS

Clases De Utilidad

Utilidades De Espaciado Y Alineación De Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Colores Y Fondo De Tailwind Css

Tailwind CSS

Clases De Utilidad

Clases De Bordes De Tailwind Css

Tailwind CSS

Clases De Utilidad

Hover, Focus Y Estado De Tailwind Css

Tailwind CSS

Clases De Utilidad

Transiciones Y Animaciones De Tailwind Css

Tailwind CSS

Clases De Utilidad

Contenedores Y Columnas En Tailwind Css

Tailwind CSS

Layout

Flexbox En Tailwind Css

Tailwind CSS

Layout

Grid En Tailwind Css

Tailwind CSS

Layout

Evaluación Test Tailwind Css

Tailwind CSS

Evaluación

Evaluación Código Tailwind Css

Tailwind CSS

Evaluación

Introducción A React Y Su Ecosistema

React

Introducción Y Entorno

Instalar React Y Crear Nuevo Proyecto

React

Introducción Y Entorno

Introducción A Jsx

React

Componentes

Introducción A Componentes

React

Componentes

Componentes Funcionales

React

Componentes

Eventos En React

React

Componentes

Props Y Manejo De Datos Entre Componentes

React

Componentes

Renderizado Condicional

React

Componentes

Renderizado Iterativo Con Bucles

React

Componentes

Manejo De Clases Y Estilos

React

Componentes

Introducción A Los Hooks

React

Hooks

Estado Y Ciclo De Vida De Los Componentes

React

Hooks

Hooks Estado Y Efectos Secundarios

React

Hooks

Hooks Para Gestión De Estado Complejo Y Contexto

React

Hooks

Hooks Optimización Y Concurrencia

React

Hooks

Introducción A React Router

React

Navegación Y Enrutamiento

Definición Y Manejo De Rutas

React

Navegación Y Enrutamiento

Rutas Anidadas Y Rutas Dinámicas

React

Navegación Y Enrutamiento

Navegación Programática Redirección

React

Navegación Y Enrutamiento

Nuevos Métodos Create De React Router

React

Navegación Y Enrutamiento

Solicitudes Http Con Fetch Api

React

Interacción Http Con Backend

Solicitudes Http Con Axios

React

Interacción Http Con Backend

Estado Local Con Usestate Y Usereducer

React

Servicios Y Gestión De Estado

Estado Global Con Context Api

React

Servicios Y Gestión De Estado

Estado Global Con Redux Toolkit

React

Servicios Y Gestión De Estado

Custom Hooks Para Servicios Compartidos

React

Servicios Y Gestión De Estado

Evaluación Test React

React

Evaluación

Aprendizaje Automático

scikit-learn

Introducción Y Entorno

Introducción E Instalación

scikit-learn

Introducción Y Entorno

Introducción Al Preprocesamiento De Datos

scikit-learn

Preprocesamiento De Datos

Identificación Y Tratamiento De Valores Faltantes

scikit-learn

Preprocesamiento De Datos

Escalado De Datos

scikit-learn

Preprocesamiento De Datos

Normalización De Datos

scikit-learn

Preprocesamiento De Datos

Codificación De Variables Categóricas

scikit-learn

Preprocesamiento De Datos

Ingeniería De Características

scikit-learn

Preprocesamiento De Datos

Selección De Características

scikit-learn

Preprocesamiento De Datos

Extracción De Características

scikit-learn

Preprocesamiento De Datos

Particionamiento De Datos

scikit-learn

Preprocesamiento De Datos

Preprocesamiento De Datos Desbalanceados

scikit-learn

Preprocesamiento De Datos

Introducción A La Regresión

scikit-learn

Regresión

Regresión Lineal

scikit-learn

Regresión

Regresión Knn Kneighborsregressor

scikit-learn

Regresión

Regresión Svm Con Svr

scikit-learn

Regresión

Regresión Con Árboles Decisiontreeregressor

scikit-learn

Regresión

Regresión Con Algoritmos De Conjunto

scikit-learn

Regresión

Introducción A La Clasificación

scikit-learn

Clasificación

Clasificación Con Regresión Logística

scikit-learn

Clasificación

Clasificación Knn Kneighborsclassifier

scikit-learn

Clasificación

Clasificación Svm Con Svc

scikit-learn

Clasificación

Clasificación Con Árboles Decisiontreeclassifier

scikit-learn

Clasificación

Clasificación Con Algoritmos De Conjunto

scikit-learn

Clasificación

Reducción De La Dimensionalidad Con Pca

scikit-learn

Aprendizaje No Supervisado

Clustering Con Kmeans

scikit-learn

Aprendizaje No Supervisado

Clustering Jerárquico

scikit-learn

Aprendizaje No Supervisado

Clustering De Densidad Con Dbscan

scikit-learn

Aprendizaje No Supervisado

Preprocesamiento De Textos Para Nlp

scikit-learn

Nlp

Representación De Texto Y Extracción De Características

scikit-learn

Nlp

Clasificación De Texto Con Scikit Learn

scikit-learn

Nlp

Análisis De Sentimiento

scikit-learn

Nlp

Técnicas Avanzadas De Extracción De Características

scikit-learn

Nlp

Introducción Al Análisis De Series Temporales

scikit-learn

Series Temporales

Preprocesamiento De Datos De Series Temporales

scikit-learn

Series Temporales

Ingeniería De Características Para Series Temporales

scikit-learn

Series Temporales

Transformación Y Escalado De Series Temporales

scikit-learn

Series Temporales

Validación Y Evaluación De Modelos En Series Temporales

scikit-learn

Series Temporales

Validación Y Evaluación De Modelos

scikit-learn

Validación De Modelos

Técnicas De Validación Cruzada

scikit-learn

Validación De Modelos

Métricas De Regresión

scikit-learn

Validación De Modelos

Métricas De Clasificación

scikit-learn

Validación De Modelos

Ajuste De Hiperparámetros

scikit-learn

Validación De Modelos

Introducción A Pipelines

scikit-learn

Pipelines Y Despliegue

Creación De Pipelines Básicos

scikit-learn

Pipelines Y Despliegue

Preprocesamiento De Datos Con Pipelines

scikit-learn

Pipelines Y Despliegue

Pipelines Y Validación Cruzada

scikit-learn

Pipelines Y Despliegue

Pipelines Con Columntransformer

scikit-learn

Pipelines Y Despliegue

Exportar E Importar Pipelines

scikit-learn

Pipelines Y Despliegue

Introducción E Instalación De Opencv

OpenCV

Introducción Y Entorno

Carga Y Visualización De Imágenes

OpenCV

Manipulación Imágenes

Operaciones Básicas En Imágenes

OpenCV

Manipulación Imágenes

Detección De Bordes Y Contornos

OpenCV

Procesamiento Y Análisis

Histograma Y Ecualización

OpenCV

Procesamiento Y Análisis

Preprocesamiento Para Machine Learning

OpenCV

Aprendizaje Automático

Clasificación De Imágenes Con Ml

OpenCV

Aprendizaje Automático

Introducción A Docker

Docker

Introducción Y Entorno Docker

Instalación De Docker

Docker

Introducción Y Entorno Docker

Descargar Imágenes De Hub.docker.com

Docker

Imágenes Docker

Crear Imágenes Con Dockerfile

Docker

Imágenes Docker

Contenedores Docker

Docker

Contenedores Docker

Volúmenes Docker

Docker

Volúmenes Docker

Redes Docker

Docker

Redes Docker

Creación De Archivos Docker Compose

Docker

Docker Compose

Docker Compose Para Varios Servicios

Docker

Docker Compose

Accede GRATIS a C y certifícate

Certificados de superación de C

Supera todos los ejercicios de programación del curso de C y obtén certificados de superación para mejorar tu currículum y tu empleabilidad.

Objetivos de aprendizaje de esta lección

  1. Identificar la utilidad de las estructuras para organizar datos heterogéneos.
  2. Diferenciar entre declaración y definición de estructuras.
  3. Usar los operadores . y -> para acceder a los miembros de las estructuras.
  4. Implementar estructuras anidadas para representar jerarquías de datos complejas.
  5. Gestionar arreglos y punteros a estructuras, optimizando el uso de memoria.