Ir al contenido principal

C: Estructuras...

En esta entrada, hablaremos acerca de las estructuras (o registros) en el lenguaje C; que son conjuntos heterogéneos de datos donde cada elemento de los mismos (campos) es accesible mediante un identificador.

Antes de empezar:


Las estructuras son una de las opciones con más potencial que nos ofrece C, ya que gracias a ellas podremos realizar muchas cosas que hasta el momento no podíamos con todos los tipos de datos y sus derivados que fuimos explicando hasta el momento.


Además, constituyen una base importante sobre la que luego se desarrolló la programación orientada a objetos, generalizando este concepto en C++ con las llamadas clases como veremos a su debido tiempo. Por lo que imaginen lo importantes que son.


Definiendo una estructura:


Luego de aclarar la importancia de estas estructuras, veamos cómo definirlas para luego poder utilizarlas.

Para definirlas lo haremos arriba de la función main y utilizaremos la palabra reservada struct usando la siguiente sintaxis:


Donde el nombre de la estructura será el que elijamos y los campos (o miembros) serán los elementos que contendrá nuestra estructura.

Veamos un ejemplo:


Creamos una estructura "persona" que tendrá como campos el nombre, edad y nacionalidad de la persona en cuestión, donde los campos son dos cadenas de caracteres (nombre y nacionalidad) y un entero (edad).

 Notamos que los campos son variables que se declaran de la misma manera que ya vimos en entradas anteriores. Y que al crear la estructura estamos creando un nuevo tipo de dato estructurado.

Declarando una variable de tipo struct:


Una vez definida nuestra estructura, podemos crear una variable de este nuevo tipo de dato estructurado, para ello simplemente declararemos una variable como cualquier otra anteponiendo como tipo de dato la palabra "struct" y el nombre de la estructura.


En el ejemplo anterior, si quisiéramos crear una "persona1" simplemente colocaríamos:




Con esto se reserva el espacio de memoria necesario para almacenar una variable de nuestro tipo "struct persona" que podrá almacenar dos cadenas de caracteres de hasta 20 caracteres (en realidad 19 y el caracter nulo) y un entero.

Accediendo a los campos de una variable de tipo struct:


Para acceder a los campos de una variable de tipo struct utilizaremos el operador "." entre el identificador de nuestra variable y el del campo correspondiente. Explicado así podría parecer un poco confuso si aún no se tienen claros los conceptos, pero es bastante simple como vemos en el ejemplo siguiente:




Aquí accedemos al campo edad de nuestra variable persona1 de tipo struct y le asignamos un valor de tipo entero. Lo mismo haríamos para los demás campos.


También podemos realizar la misma asignación al declarar la variable de la manera que vemos a continuación:



O directamente podemos declarar todos los campos de persona1 en el orden correspondiente:


Arreglos de estructuras:


Como podíamos crear arrays de diferentes tipos, también podríamos crear uno donde cada elemento del array sea del tipo struct:


Y donde para acceder a los datos correspondientes a cada persona del array (nombre, edad, nacionalidad) lo haremos mediante el índice como en cualquier array pero debemos agregar el campo correspondiente luego del operador punto:


Copiando una variable de tipo struct:


A diferencia de los arrays donde no podíamos hacer una copia mediante asignación directa (sino que debíamos copiar cada elemento) las estructuras sí nos permiten hacerlo, por lo que si colocamos:




La variable persona2 tendría todos los campos iguales que persona1.


Punteros a estructuras:


Para terminar, también podemos definir punteros a estructuras de la misma manera que hacíamos con otros tipos de datos.


Veamos cómo creando un puntero a nuestra variable persona1 de tipo struct:


Hasta aquí no hay diferencia con lo que ya conocíamos, pero si queremos acceder al contenido de alguno de los campos de nuestra variable de tipo struct a través del puntero utilizaremos un nuevo operador que se conoce como operador flecha (->).

Entonces si quisiéramos por ejemplo acceder mediante el puntero al nombre de persona1 haríamos:


Si queremos acceder a los campos mediante el operador punto como hacíamos hasta ahora pero usando el puntero, haríamos:








En las próximas entradas seguiremos avanzando en el lenguaje C y también colocaremos varios ejemplos típicos y algoritmos famosos explicados paso a paso donde usaremos todo lo aprendido.

Comentarios

Entradas populares de este blog

C: Conversiones de tipo (casting) en C...

El casting o simplemente cast  nos permite hacer una conversión explícita de un tipo de dato a otro, a criterio del programador siempre y cuando estos tipos sean compatibles. Este cast se realiza a través de un operador de conversión de tipos (type casting operator) y es un recurso a tener en cuenta ya que hay situaciones en que nos puede resultar de gran utilidad. Hacer uso de un cast es tan sencillo como poner (tipo de dato)  delante de la expresión o variable a convertir. Veamos un ejemplo: Declaramos una variable de tipo int con un identificador tan creativo como "a" y le realizamos diferentes cast a a para mostrarlo como si fuera un float, un double y un char en un printf. Lo que obtendríamos en pantalla sería lo siguiente: Donde tenemos el valor de nuestro a, a convertido en float y double (mostrándolo con 3 cifras decimales) y a convertido en char. Si vemos este último caso, al hacer la conversión de "a" a char toma a como el código ascii de

C: Ejemplos: Congruencia de Zeller (nivel básico) ...

La Congruencia de Zeller es un algoritmo que se atribuye al matemático alemán Julius Christian Johannes Zeller que vivió en el siglo XIX. Este algoritmo nos permite determinar el día de la semana que le corresponde a una fecha determinada del calendario Gregoriano. La fórmula que nosotros usaremos (con algunas modificaciones respecto de la original para poder usarla en  informática) es la siguiente: Donde h es el día de la semana (entre 0 y 6), J es año/100 (la centuria) y K es año mod 100 (el año de la centuria). Y hay que tener en cuenta que los meses de enero y febrero cuentan como el mes 13 y 14 del año anterior. Ahora que tenemos la fórmula, programemos el algoritmo en C mediante el uso de una función: Analicemos el código paso a paso: Tenemos en cuenta el caso de enero y febrero: Dijimos que estos meses corresponden a los meses 13 y 14 del año anterior por lo que los asignamos como corresponde (mes + 12 , que dará 13 para enero y 14 para febrero) y le rest

Algoritmos: Resolución de problemas y refinamientos en pseudocódigo...

En otras entradas, vimos las partes que debe tener nuestro algoritmo en pseudocódigo y las estructuras que utilizaremos para resolverlo. Ahora llega el turno de implementar todo en conjunto para dar origen a nuestra creación. Pero ¿cómo resolvemos un problema así? Para hacerlo, utilizaremos lo que llamamos refinamientos sucesivos. Este concepto consiste en dividir el problema en subproblemas más pequeños y a estos, a su vez, en otros más pequeños; y así sucesivamente hasta que la solución de los últimos sea trivial, sencillo de resolver. Luego usaremos todas las soluciones obtenidas para armar la solución de nuestro problema mayor. Este principio, tiene base en parte de la técnica divide and conquer (dependiendo de la traducción: "divide y vencerás") que es una de las muchas técnicas de resolución de algoritmos existentes. Como vemos, al dividir el problema en otros más pequeños y más fáciles de resolver, podemos pasar de un problema complicado a uno cuya solución es much