Punteros

Descargar como pdf o txt
Descargar como pdf o txt
Está en la página 1de 8

Apuntes de Punteros en ANSI C Escuela Superior de Informtica Ciudad Real (UCLM) 1.

Tabla de Contenidos

INTRODUCCIN ...................................................................1
QU ES UN PUNTERO?.....................................................1
PARA QU USAR PUNTEROS?........................................2
ARITMTICA DE PUNTEROS ..............................................3
CADENAS DE CARACTERES..............................................4
ESTRUCTURAS....................................................................5
PUNTEROS A MATRICES....................................................5
ASIGNACIN DINMICA DE MEMORIA.............................6
PUNTEROS A FUNCIONES..................................................7
Y AHORA QU?..................................................................8

Introduccin
Djole el perro al hueso: Si t eres duro,
yo tengo tiempo... Refranero Popular

uele resultar difcil dominar el concepto de punte-
ro. Especialmente a los programadores que
aprendieron con lenguajes de alto nivel, donde no se
permite trabajar directamente con posiciones de
memoria. El mayor problema en C surge al tratar de
detectar esos segmentation fault maquiavlicos
que a veces aparecen, a veces no.

Este documento pretende llenar esta laguna con un
texto breve, que introduce los fundamentos de los
punteros con numerosos ejemplos. No es posible
aprender a programar nicamente leyendo un docu-
mento, as que aplicando el Caminante no hay
camino, se hace camino al andar, utiliza un compila-
dor ANSI C y compila, prueba y cambia los ejemplos
de esta gua. Si nicamente lees el texto, no cogers
manejo y tu lectura valdr para poco.

Miles de agradecimientos a nuestros compaeros
Miguel ngel Laguna, Ramn Manjavacas y David
Villa por aportar ideas y depurar este documento.

Ser bienvenida cualquier indicacin acerca de este
texto, errores encontrados, dudas y aclaraciones en
nuestra cuenta de e-mail.
Qu es un puntero?
En el famoso libro de Kernighan y Ritchie El lenguaje
de programacin C, se define un puntero como una
variable que contiene la direccin de una variable.
Tras esta definicin clara y precisa, podemos remar-
car que un puntero es un tipo especial de variable
que almacena el valor de una direccin de memoria.

La memoria de los ordenadores est formada por un
conjunto de bytes. Simplificando, cada byte tiene un
nmero asociado; una direccin en esa memoria.

145 146 147 148 149 150 151 152 153 154
... ...
Direcciones de Memoria
Contenido de cada
Posicin de Memoria

Figura 1. Representacin de la Memoria
La representacin anterior se utilizar a lo largo del
texto para ilustrar los distintos ejemplos. De esta
forma, si ejecutamos el ejemplo del Listado 1:

01 #include <stdio.h>
02 int main(void)
03 {
04 float pi=3.14;
05 printf ("%.2f\n", pi);
06 return 0;
07 }
Listado 1
Obtenemos en pantalla el esperado 3.14. Sin embar-
go, qu operaciones ha hecho el ordenador hasta
mostrar el resultado?. En la lnea 4, la instruccin
reserva memoria para la variable pi. En este ejemplo,
asumimos que un decimal de tipo float ocupa 4 bytes.
Dependiendo de la arquitectura del computador, la
cantidad de bytes requerida para cada tipo de datos
puede variar.

La Figura 2 muestra el estado de la memoria despus
de la declaracin de la variable pi. La representacin
elegida facilita el seguimiento de los ejemplos, sin
embargo, la representacin real en la memoria de
ordenador no tendra esa disposicin.
S

2003 Carlos Gonzlez Morcillo - Miguel ngel Redondo Duque. Se permite la copia, distribucin y/o modificacin de este documento bajo los trminos de la
licencia de documentacin libre GNU, versin 1.1 o cualquier versin posterior publicada por la Free Software Foundation; sin secciones invariantes. Puede
consultar esta licencia en http://www.gnu.org
Carlos Gonzlez Morcillo ::: [email protected]
Miguel ngel Redondo Duque ::: [email protected]
http://www.inf-cr.uclm.es/www/cglez
http://www.inf-cr.uclm.es/www/mredondo

Punteros en C
Escuela Superior de Informtica - Ciudad Real
Universidad de Castilla-La Mancha

ltima Revisin: 28 de octubre de 2003 ::.
.2 Carlos Gonzlez Morcillo ([email protected]) ::: Miguel ngel Redondo Duque ([email protected])
145 146 147 148 149 150 151 152 153 154
... ... 3 . 1 4
Direccin de comienzo de pi
pi

Figura 2. Memoria reservada para una variable
Cuando se utiliza pi en la lnea 5, ocurren dos pasos
diferenciables. En primer lugar, el programa busca la
direccin de memoria reservada para pi (en nuestro
ejemplo, sera la direccin 146). Hecho esto, en
segundo lugar, el programa recupera el contenido de
esa direccin de memoria.

As, distinguimos por un lado la direccin de memoria
asignada a una variable, y por otro lado el contenido
de la posicin de memoria reservada. Podemos
acceder a la direccin de una variable utilizando el
operador &. Mediante este operador, obtenemos la
direccin donde se almacena la variable (un nme-
ro!). En el ejemplo del Listado 2, se ha utilizado %u y
una conversin forzada a entero sin signo para que la
salida por pantalla sea ms entendible. Se poda
haber utilizado el conversor estndar para punteros
%p y haber quitado la conversin forzada a entero sin
signo.

01 #include <stdio.h>
02 int main(void)
03 {
04 float pi=3.14;
05 printf ("Direccin de pi: %u\n",
(unsigned int)&pi);
06 return 0;
07 }
Listado 2
Si ejecutramos el programa del listado 2 en la me-
moria del ejemplo anterior, debera mostrarnos la
direccin 146. Recordemos que una direccin de
memoria es nicamente un nmero. De hecho, po-
dramos almacenarla en una variable entera sin signo
(ver Listado 3).

01 #include <stdio.h>
02 int main(void)
03 {
04 float pi=3.14;
05 unsigned int dir=(unsigned int)&pi;
06 printf ("Direcc. de pi: %u\n", dir);
07 return 0;
08 }
Listado 3
El cdigo anterior demuestra que no hay nada de
magia ni efectos especiales en el tema de las direc-
ciones. Simplemente son nmeros que pueden ser
almacenados en variables enteras
1
. Esto se puede ver
grficamente en la siguiente figura (suponiendo que
las variables de tipo entero sin signo requieran 3
bytes en memoria).

1
Recordemos que el modificador unsigned simplemente
indica que no se permiten nmeros negativos. Las direccio-
nes no pueden ser negativas.
145 146 147 148 149 150 151 152 153 154
... ... 3 . 1 4 1 4 6
pi dir
dir = &pi

Figura 3. Uso del operador '&'
Lo que se quiere representar simplemente es que
mediante el operador & accedemos a la direccin de
memoria de una variable. Esta direccin es un simple
nmero entero con el que podemos trabajar directa-
mente.

Para diferenciar una variable entera que almacene
valores de nuestro programa (por ejemplo, ndices,
contadores, valores...) de las variables que almace-
nan direcciones, en C se cre un nuevo tipo de varia-
ble para almacenar direcciones. Este tipo es, preci-
samente, el puntero. As, podemos tener algunas
declaraciones de punteros como las siguientes:

char * nombre;
float * altura;

Cmo interpretar las lneas anteriores? Podemos leer
que nombre es un entero, pero podemos aadir que
es un entero especialmente diseado para guardar la
direccin de un carcter. Por su parte, altura es un
entero, adems es un entero especialmente diseado
para guardar la direccin de un float. Es lo que deci-
mos resumidamente con nombre es un puntero a
char o altura es un puntero a float.

El operador * tambin se utiliza para recuperar el
contenido de una posicin de memoria identificada
por una direccin. No debemos confundir su uso en la
declaracin de un puntero, como hemos visto antes,
con los casos en que se emplea para recuperar el
contenido de una posicin de memoria. Este uso lo
veremos a continuacin.
Para qu usar punteros?
Hagamos un programa que cambie el valor de la
variable pi del ejemplo anterior a cero. Para ello,
realizaremos el cdigo que se muestra en el Listado
4. En este caso intentamos pasar el parmetro vpi
por referencia, para que cambie el contenido de pi
dentro de la funcin. Si ejecutamos el programa, el
resultado obtenido no es el esperado. Por qu?

01 void cambiavalor(float vpi) {
02 vpi = 0;
03 printf ("Direccin de vpi: %u\n",
(unsigned int) &vpi);
04 printf ("Valor de vpi: %f\n", vpi);
05 }
06 int main(void){
07 float pi=3.14;
08 cambiavalor(pi);
09 printf ("Direccin de pi: %u\n",
(unsigned int) &pi);
10 printf ("Valor de pi: %f\n", pi);
11 return 0;
14 }
Listado 4
Apuntes de Punteros en ANSI C Escuela Superior de Informtica Ciudad Real (UCLM) 3.
La ejecucin del programa muestra que el parmetro
pi ha sido pasado por valor (en lugar de por referen-
cia). Si hacemos un seguimiento ms exhaustivo del
programa, vemos que cuando llamamos a la funcin
cambiavalor en la lnea 8, el programa salta a la
lnea 1 para ejecutar la funcin. All, se crea una
nueva variable llamada vpi, y se copia el valor de pi
en ella. Esto se representa en el esquema de la
Figura 4.
145 146 147 148 149 150 151 152 153 154
... ... 3 . 1 4 3 . 1 4
pi vpi
cambiavalor (pi);

Figura 4. Copia del valor de pi en vpi
Queda clara la razn por la que vpi y pi tienen
diferente direccin en memoria; se ha realizado una
copia del contenido de la memoria que tiene pi, se ha
llevado a otra zona de memoria y se le ha puesto de
nombre vpi. De esta forma, cuando se asigna 0 a la
variable vpi en la lnea 2, se est asignando al conte-
nido de memoria de vpi, no al de pi (ver Figura 5).
145 146 147 148 149 150 151 152 153 154
... ... 3 . 1 4 0 . 0 0
pi vpi
vpi = 0;

Figura 5. Cambio de valor de vpi
Los argumentos de las funciones en C se pasan
siempre por valor. No existe una forma directa para
que la funcin que se invoca altere una variable de la
funcin que la llama. Es imprescindible el uso de
punteros en el caso de querer modificar el valor de
una variable pasada como parmetro a una funcin.

As, podemos modificar el programa del Listado 4,
para que el paso del argumento se realice por refe-
rencia. Tendremos que utilizar los operadores * y &
que hemos visto anteriormente. El nuevo programa
se muestra en el Listado 5.

00 #include <stdio.h>
01 void cambiavalor(float *vpi) {
02 *vpi = 0;
03 printf ("Dir. vpi: %u\n",
(unsigned int) vpi);
04 printf ("Val. vpi: %f\n", *vpi);
05 }
06 int main(void) {
07 float pi=3.14;
08 cambiavalor(&pi);
09 printf ("Dir. pi: %u\n",
(unsigned int)&pi);
10 printf ("Val. pi: %f\n", pi);
11 return 0;
12 }
Listado 5
En esta nueva versin del programa, la funcin cam-
biavalor, recibe el parmetro como un puntero
(lnea 1). Le estamos diciendo al compilador que pase
como argumento a la funcin una direccin de memo-
ria. Es decir, queremos que le pase un entero; ese
entero ser una direccin de memoria de un float.

En un seguimiento en detalle del programa vemos
que, en la lnea 8, cuando llamamos a la funcin
cambiavalor, le tenemos que pasar una direccin de
memoria. Para ello, tenemos que hacer uso del ope-
rador &, para pasarle la direccin de la variable pi.
Con esto, cuando llamamos a la funcin cambiavalor
en 1, lo que se hace es una copia de la direccin de
memoria de pi en la variable vpi (ver Figura 6). As,
mediante el operador * podremos acceder al conteni-
do de la posicin de memoria de pi y cambiar su
valor.
145 146 147 148 149 150 151 152 153 154
... ... 3 . 1 4 1 4 6
pi vpi
cambiavalor(&pi);

Figura 6. Paso por referencia de "pi"
En la lnea 2 del programa se cambia el valor de la
zona de memoria apuntada por vpi. Podemos leer
esa instruccin como asignamos cero al contenido de
la direccin de memoria apuntada por vpi. Recorde-
mos que esa zona de memoria tiene que ser de tipo
float. En el ejemplo de la Figura 6, esta direccin de
memoria es la 146. Podemos ver un esquema del
resultado de la operacin en la Figura 7.
145 146 147 148 149 150 151 152 153 154
... ... 0 . 0 0 1 4 6
pi vpi
*vpi = 0;

Figura 7. Asignacin de 0 al contenido de memoria
apuntado por vpi
Ahora, cuando acabe la funcin cambiavalor y retor-
nemos a la funcin principal main, el valor de pi ser
0. Hemos realizado un paso de parmetro por refe-
rencia.

A continuacin examinaremos en profundidad algunos
tipos de datos complejos (como vectores y estructu-
ras) y su utilizacin con punteros, as como algunos
operadores nuevos especficos de punteros.
Aritmtica de Punteros
Como hemos mencionado anteriormente, podemos
considerar el tipo de un puntero como el tipo de datos
que contiene la zona de memoria a la que ste apun-
ta. Es decir, si tenemos:

int *ptr;
*ptr = 2;

la primera lnea indica que tenemos una variable
llamada ptr, cuyo contenido es una direccin de
memoria a una zona donde hay un entero. En la
segunda lnea asignamos al contenido de esa direc-
cin de memoria el valor 2.

Una operacin interesante que podemos realizar con
punteros es incrementarlos. Obviamente, la operacin
de incremento en un puntero no es igual que en una
variable normal. Un incremento en un puntero har
.4 Carlos Gonzlez Morcillo ([email protected]) ::: Miguel ngel Redondo Duque ([email protected])
que apunte al siguiente elemento del mismo tipo de
datos que se encuentre en memoria.

En el ejemplo de la Figura 8, inicialmente el puntero
ptr apunta al tercer entero almacenado en el bloque
de memoria representado (suponiendo que cada
entero ocupa 3 posiciones de memoria). As, la asig-
nacin *ptr=168 har que el contenido de esa posi-
cin de memoria sea 168. Si incrementamos el
puntero (cuidado!, no hablamos del contenido del
puntero, sino del propio puntero), haremos que
apunte al siguiente entero que hay en memoria. Este
incremento del puntero no implica sumar 1 a la
direccin que contiene el puntero. En este ejemplo,
sera necesario sumar 3 (tres posiciones de memoria
que ocupa el entero para acceder al siguiente).

Naturalmente esta suma de posiciones de memoria
no la tendr que hacer el programador. Ser el com-
pilador el que se encargue de realizar las cuentas
para ver cuntas posiciones de memoria debe avan-
zar. En general, el compilador tendr que sumar
tantas posiciones de memoria como requiera el tipo
de datos sobre el que se haya definido el puntero.

0
...
1 2 0 0 5 1 6 8
0 0 2 7 4 0 4 9 1
0 6 6 3 1 8 4 2 3
7 7 1 0 0 3 8 0 7 ...
Cada entero ocupa 3 posiciones de memoria
*ptr=168;
ptr++;

Figura 8. Incremento de punteros

En el caso de que incrementemos alegremente el
puntero y nos salgamos de la zona de memoria que
tengamos reservada, recibiremos la visita de nuestro
amigo segmentation fault.

Vamos a ilustrar el uso de punteros con vectores
mediante un ejemplo. Supongamos que definimos un
array de enteros (que, a fin de cuentas, es un conjun-
to de enteros a los que se reservan posiciones de
memoria consecutivas) como se muestra en la lnea 2
del Listado 6.

01 #include <stdio.h>

02 int vector[] = {1, 56, 47, -89, 10};
03 int *ptr;

04 int main(void)
05 {
06 int i;
07 ptr = &vector[0];
08 for (i=0; i<5; i++) {
09 printf ("Vect[%d] = %d\n", i, vector[i]);
10 printf ("Ptr + %d = %d\n", i, *(ptr+i));
11 }
12 return 0;
13 }
Listado 6
Tenemos un vector de 5 enteros. Podemos acceder a
ellos mediante el ndice que ocupan en el array (como
por ejemplo, vector[3] nos dara el cuarto entero del
array). De igual forma podemos acceder a l median-
te un puntero. En la lnea 7, asignamos al puntero la
direccin del primer elemento del vector. En C, el
nombre del array apunta al primer elemento del
array; por lo que podramos cambiar la lnea 7 por la
instruccin ptr=vector; con idnticos resultados.

De cualquier forma, debemos tener muy claro que,
una vez definido un array, el nombre del array identi-
fica a la posicin del primer elemento, pero es una
constante. Es decir, podemos utilizarlo como hemos
indicado antes para asignrselo a un puntero (copia-
mos la direccin de memoria del primer elemento),
pero no podemos asignarle a un array ya creado una
direccin distinta; vector = otro_puntero provoca-
ra un error. Algunos programadores llaman al nom-
bre de un array como puntero constante, indicando
que no puede cambiarse su valor. Es simplemente un
puntero al que no podemos cambiar su direccin.
Cadenas de caracteres
Recordemos que en C, las cadenas de caracteres son
simplemente arrays de caracteres. No hay un tipo de
datos especial, como en Basic o Pascal. Una cadena
en C terminar siempre con un carcter especial
reservado para indicar fin de cadena, que es un
cero binario, representado por \0.

Vamos a implementar nuestra funcin de copia de
cadenas, similar a la de ANSI C: strcpy().

01 #include <stdio.h>

02 char origen[40] = "Apuntes de Punteros";
03 char destino[40];

04 char *mi_strcpy(char *dest, const char *orig)
05 {
06 char *p = dest;
07 while (*orig != '\0')
08 *p++ = *orig++;
09 *p = '\0';
10 return dest;
11 }

12 int main(void)
13 {
14 int i;

15 mi_strcpy(destino, origen);
16 printf ("%s", destino);
17 return 0;
18 }
Listado 7
En el listado anterior, el espacio necesario para las
dos cadenas se reserva en origen y destino. Cuando
llamamos a la funcin mi_strcpy, en la lnea 15 lo
nico que se pasa es la direccin de memoria de
ambas cadenas. Este planteamiento nos permite
trabajar con arrays de gran tamao sin tener que
mover gran cantidad de informacin en memoria.
Como las cadenas de caracteres son exactamente
iguales que los arrays de cualquier otro tipo de datos,
el enfoque utilizado es el mismo en todos los casos.

Recordemos que (ver Figura 6), lo que pasamos como
argumento a la funcin es una copia de la direccin
de memoria. As, si cambiamos el puntero dentro de
la funcin mi_strcpy, a la vuelta de la funcin los
punteros originales seguirn apuntando a donde
estaban. Esto se puede ver en el ejemplo anterior;
cuando llamamos a la funcin en la lnea 15, los
punteros origen y destino apuntan al comienzo de
las cadenas. Dentro de la funcin mi_strcpy, los
Apuntes de Punteros en ANSI C Escuela Superior de Informtica Ciudad Real (UCLM) 5.
punteros van avanzando en la cadena (lnea 8), y al
final ambos estn situados al final de las cadenas. Al
ser copias, cuando volvemos de la funcin, los punte-
ros origen y destino siguen apuntando al inicio de la
cadena.

El modificador const usado en la lnea 4 hace que,
dentro de la funcin, no podamos modificar el conte-
nido de la memoria apuntada por orig, tomando el
contenido como valor constante.

En la lnea 6, hacemos que un puntero auxiliar apunte
al inicio de la cadena destino (dest). Despus, mien-
tras el carcter apuntado por el puntero origen (orig)
sea distinto del fin de cadena, copiamos el carcter
en la zona de memoria apuntada por p y avanzamos.
Esto lo realizamos con una nica instruccin (lnea 8).
Es equivalente *p++ a (*p)++?. No, segn la prece-
dencia de los operadores, *p++ es equivalente a
*(p++). Y, qu decimos en cada caso?. Pues con la
primera instruccin, *p++, decimos que primero
veamos el contenido del puntero y despus
incrementemos la posicin del puntero (utilizado en la
funcin de copia de arrays), mientras que con (*p)++
decimos que incremente el contenido de la posicin
de memoria donde apunta p. Si p apuntara a un
carcter C, hara que ese carcter pasara a ser D.
Si p apunta a un entero cuyo valor es 24, despus de
esa instruccin sera 25.

Queda claro, por tanto, la necesidad de establecer
correctamente el orden de aplicacin entre operado-
res. Personalmente creo una prdida de tiempo tratar
de recordar la precedencia de los operadores. Una
buena prctica que mantiene el cdigo ms legible es
utilizar parntesis siempre que haya dudas. Natural-
mente, siempre dentro de unos lmites razonables.

Es necesario el puntero auxiliar p? Claramente no,
se ha utilizado nicamente para dejar bien claro que
trabajamos con direcciones de memoria, y que *p y
*dest es exactamente lo mismo. Por eso, al retornar
de la funcin, devolvemos dest. El cdigo de la
funcin mi_strcpy del Listado 7 poda simplificarse
como muestra el Listado 8.

01 while (*orig != '\0')
02 *(dest++) = *(orig++);
03 *dest = '\0';
04 return dest;
Listado 8
En esta ltima versin se ha hecho uso de parntesis
para indicar de forma clara, que queremos copiar el
contenido de la memoria apuntada por orig en dest
y, despus, incrementar los punteros. Finalmente hay
que aadir la constante \0 en la ltima posicin de
dest (para que cumpla el convenio utilizado con
cadenas de caracteres en ANSI C). Incluso podramos
optimizar algo ms el cdigo poniendo en la definicin
del bucle while(*orig), ya que ser cierto mientras
el valor apuntado por orig sea distinto de cero.

Finalmente recordemos que podemos utilizar tambin
la notacin de array para trabajar con cadenas de
caracteres. As, tendramos las expresiones dest[i] y
*(dest + i) como equivalentes.
Estructuras
Una estructura es un bloque que contiene una se-
cuencia de variables de diferentes tipos de datos.
Vamos a realizar un programa que muestre por
pantalla un elemento de tipo estructura.

01 #include <stdio.h>
02 #include <string.h>

03 struct alumno{
04 char nombre[20];
05 char apell[40];
06 int edad;
07 float nota;
08 };

09 void ver_alumno (struct alumno *a) {
10 printf ("%s %s ", a->nombre, a->apell);
11 printf ("Calificacion: %f\n", a->nota);
12 }

13 int main(void)
14 {
15 struct alumno alum;
16 strcpy (alum.nombre, "Perico");
17 strcpy (alum.apell, "Palotes");
18 alum.edad = 22;
19 alum.nota = 8.75;
20 ver_alumno (&alum);

21 return 0;
22 }
Listado 9
Un error tpico es olvidar el punto y coma al terminar
la definicin de una estructura (lnea 8). Despus de
declarar una variable del tipo estructura alumno,
dentro de main, podemos acceder a sus campos con
el operador punto siempre que estemos trabajando
con la estructura directamente. Cuando estemos
utilizando un puntero a estructura, utilizaremos el
operador flecha (->). Este operador es equivalente a
utilizar (*puntero_estructura).campo

As, en la lnea 11 podamos haber accedido al campo
nota del puntero a estructura a, mediante la ins-
truccin (*a).nota. Sin embargo, es ms cmodo
utilizar la notacin flecha para estos casos.

ANSI C permite pasar por valor las estructuras com-
pletas. Sin embargo, esto implica que se copia el
contenido de toda la estructura (completa) desde la
funcin llamante a la funcin llamada. Esto, para
estructuras con un gran nmero de elementos, puede
ser un problema. Por cuestiones de eficiencia y co-
modidad, pasaremos las estructuras a las funciones
mediante un puntero.
Punteros a Matrices
Anteriormente hemos visto la relacin entre punteros
y arrays; vamos a ampliar nuestro campo de visin
utilizando arrays de varias dimensiones (matrices).

01 #include <stdio.h>
02 #define NFILAS 5
03 #define NCOLUMNAS 8

04 int main(void)
05 {
06 int fil, col;
07 int matriz[NFILAS][NCOLUMNAS];

08 /* Asignamos valores a los elementos */
.6 Carlos Gonzlez Morcillo ([email protected]) ::: Miguel ngel Redondo Duque ([email protected])
09 for (fil=0; fil<NFILAS; fil++)
10 for (col=0; col<NCOLUMNAS; col++)
11 matriz[fil][col] = fil*col;

12 /* Mostramos los valores de 2 formas */
13 for (fil=0; fil<NFILAS; fil++)
14 for (col=0; col<NCOLUMNAS; col++){
15 printf ("%d - ", matriz[fil][col]);
16 printf ("%d\n", *(*(matriz+fil)+col));
17 }
18 return 0;
19 }
Listado 10
En el ejemplo del Listado 10, tenemos una declara-
cin de matriz en la lnea 7. Qu significa esa ins-
truccin? Por un lado tenemos un array de 5 elemen-
tos (filas), pero a su vez cada uno de esos elementos
son arrays de 8 enteros (columnas).

Sabemos que los arrays se almacenan contiguos en
memoria. As, el estado de nuestra memoria podra
representarse como:

000000000123456702468101214036912151821...

El primer cero indica la direccin de &matriz[0][0]
(que podemos nombrar tambin nicamente con el
nombre matriz). De esta forma, si quisiramos
mostrar el tercer elemento de la segunda fila (el
nmero 2 marcado en negrita), podramos hacerlo
como matriz[1][2] o bien como *(*(matriz+1)+2).
La primera forma es sencilla; utilizamos la notacin
de array a la que estamos acostumbrados. La nota-
cin de punteros podemos leerla como: Dame el
contenido de una direccin. Esa direccin est forma-
da por el contenido de donde apunta matriz ms uno.
A ese resultado smale dos. Podemos ver cada fila
de la matriz como un puntero cuyos elementos son
punteros a arrays de enteros (ver Figura 9).

int matriz[5][8];
1 2 3 4 5 6 7 8
9 10 1112 ..
En memoria
principal...
123456789101112...

Figura 9. Representacin de matriz bidimensional
As, podemos acceder al elemento de la segunda fila,
segunda columna pidiendo primero el contenido de
matriz+1. Esto nos dar el segundo puntero del array
(recordemos que los arrays en C empiezan a nume-
rarse en 0). Con esto, tenemos una direccin de
memoria. Si a esa direccin de memoria le sumamos
dos, accedemos al tercer elemento del segundo array
(en el ejemplo de la Figura 9, tendramos el 11). En
caso de querer acceder al primer elemento del se-
gundo array, bastara con escribir *(*(matriz+1)) o,
de forma equivalente **(matriz+1).

Queda claro, por tanto, que el primer componente de
la matriz bidimensional es equivalente a un array del
siguiente tipo: int *filas[NFILAS];. En esta decla-
racin estamos diciendo que tenemos un vector cuyos
elementos son punteros a enteros, lo mismo que
tenamos en la parte de filas de la declaracin ante-
rior.

Qu tiene que hacer el compilador para acceder a
cada uno de los elementos de la matriz? Debe calcu-
lar la posicin del elemento en memoria. Para ello,
deber multiplicar la fila en la que se encuentra por el
nmero de elementos por columna y sumarle la
columna en la que est actualmente. Es decir:

Posicin en Memoria = (fila * NCOLS) + columna

Como puede verse en la expresin anterior, es nece-
sario que el compilador sepa a priori, el nmero de
elementos que tiene cada columna de nuestra matriz.
En general, es necesario especificar, al menos, la
segunda dimensin de la matriz de forma fija. Puede
interesarnos implementar una funcin que nos permi-
ta trabajar con nuestro array multidimensional. Un
ejemplo muy sencillo (aunque intil) de esto puede
verse en el Listado 11.

01 void reset_matriz(int mat[][NCOLUMNAS])
02 {
03 int fil, col;
04 for (fil=0; fil<NFILAS; fil++)
05 for (col=0; col<NCOLUMNAS; col++)
06 mat[fil][col] = 1;
07 }
Listado 11
En general, en arrays de ms de una dimensin es
necesario indicar, en la definicin de los parmetros a
la funcin, las dimensiones segunda, tercera, etc...
Asignacin dinmica de memoria
En muchas ocasiones no podemos saber a priori el
espacio que vamos a necesitar para nuestras estruc-
turas de datos. Esto deber decidirse en tiempo de
ejecucin y llevarse a cabo mediante funciones de
gestin de memoria como malloc, calloc, u otras
similares.

Cuando pedimos memoria con alguna funcin de
gestin (como malloc()), sta nos devuelve un punte-
ro. En compiladores ANSI, este puntero es de tipo
void. Como void es un tipo genrico que apunta a
cualquier cosa, tendremos que indicar al compilador
que haga una conversin de tipo (cast) al tipo de
datos que nos interese; es decir, a un puntero al tipo
de datos que queramos.

Adems, ser necesario indicarle a la funcin que
llamemos para pedir memoria el nmero de bytes
que necesitamos. En C, el nmero de bytes que se
necesitan para almacenar cada tipo de datos es
dependiente de la mquina sobre la que ejecutemos
el programa. Por ejemplo, en dos mquinas distintas
una puede requerir 2 bytes para un entero y otra
utilizar 4 bytes para el mismo entero. Cmo pode-
mos decirle el nmero de bytes que vamos a necesi-
tar? Utilizando la funcin sizeof(). As, el tamao
que queremos reservar estar en funcin del tamao
del tipo de datos que utilicemos. Esto puede verse en
el ejemplo del Listado 12.
Apuntes de Punteros en ANSI C Escuela Superior de Informtica Ciudad Real (UCLM) 7.

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

03 int main(void)
04 {
05 int *ptr, i;

06 ptr = (int *)malloc(10*sizeof(int));
07 if (ptr==NULL) printf ("Error de Mem.");
08 for (i=0; i<10; i++)
09 ptr[i]=1;
10 return 0;
11 }
Listado 12
Utilizando la equivalencia entre la notacin de punte-
ros y de arrays, en la lnea 9 asignamos a todos los
elementos del array creado anteriormente un valor
constante 1.

El ejemplo anterior permite crear un array de una
dimensin dinmico en tiempo de ejecucin, pero
cmo crearamos un array multidimensional?. Hemos
visto en el punto anterior que los arrays bidimensio-
nales se gestionan como arrays de punteros a arrays
del tipo de datos que queremos utilizar. Cmo reser-
vamos memoria para estas matrices?. En un plan-
teamiento general, necesitaremos generar la matriz
dinmicamente, tanto el nmero de filas como de
columnas. En los siguientes ejemplos utilizaremos
como tipo de datos base el entero, pero podra utili-
zarse cualquier otro.

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

03 int main(void)
04 {
05 int nfilas=5, ncols=10;
06 int fila;
07 int **filaptr;

08 filaptr = malloc(nfilas * sizeof(int *));
09 if (filaptr == NULL) printf ("Error");

10 for (fila=0; fila<nfilas; fila++) {
11 filaptr[fila] =
(int *) malloc(ncols * sizeof(int));
12 if (filaptr == NULL) printf ("Error");
13 }
14 return 0;
15 }
Listado 13
En el Listado 13, filaptr es un puntero a puntero a
entero. En este caso, el programa crea una matriz de
5 filas por 10 columnas. Esos datos, naturalmente,
podran gestionarse en tiempo de ejecucin.

La primera llamada a malloc (lnea 8) reserva espa-
cio para el array de punteros a los arrays de enteros.
Por esta razn, a la funcin sizeof le pasamos el tipo
puntero a entero. Con el primer array de punteros
creado, pasamos a reservar memoria para los arrays
que son apuntados por cada elemento fila (en la
lnea 11).

En el ejemplo anterior se necesitan en total 6 llama-
das a malloc; una para reservar el array de filas y
una para cada array de enteros asociado a cada fila.
Adems, no nos aseguramos de tener toda la matriz
en un bloque contiguo de memoria. Se puede com-
probar, utilizando un puntero auxiliar al primer ele-
mento de la primera fila (es decir, usando int
*ptr_aux que apunte a filaptr[0] e incrementando
*(testptr++) no accedemos a los valores introduci-
dos en la matriz). Tal vez obtenemos un bonito
segmentantion fault?.

Con este planteamiento, deberemos acceder a los
elementos de la matriz nicamente utilizando la
notacin de corchetes. No podemos utilizar un punte-
ro para recorrer la memoria.

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

03 int main(void)
04 {
05 int nfilas=5, ncols=10;
06 int *mem_matriz;
07 int **filaptr;
08 int *testptr;
09 int fila, col;
10 mem_matriz =
malloc(nfilas * ncols * sizeof(int));
11 if (mem_matriz == NULL) printf ("Error");

12 filaptr = malloc(nfilas * sizeof(int *));
13 if (filaptr == NULL) printf ("Error");

14 for (fila=0; fila<nfilas; fila++)
15 filaptr[fila] = mem_matriz+(fila*ncols);

16 for (fila=0; fila<nfilas; fila++) {
17 for (col=0; col<ncols; col++)
18 filaptr[fila][col] = 1;
19 }

20 testptr = filaptr[0];
21 for (fila=0; fila<nfilas; fila++)
22 for (col=0; col<ncols; col++)
23 printf ("[%d][%d]%d \n",
fila, col, *(testptr++));
24 return 0;
25 }
Listado 14
En el ejemplo del Listado 14 se reserva la memoria
para la matriz en un bloque contiguo. En la lnea 10
se reserva la memoria para todas las celdas de la
matriz. En la lnea 12 hacemos el array de punteros
correspondiente a las filas de la matriz. En las lneas
14-15 hacemos que cada puntero de las filas apunte
a la celda donde empieza cada fila. Esto puede calcu-
larse con un simple producto (fila*ncols).

Las lneas siguientes (16..23) demuestran que el
bloque creado es contiguo, y que podemos utilizar un
puntero auxiliar para recorrer las posiciones de me-
moria.

Con este mtodo se han tenido que utilizar nicamen-
te dos llamadas a la funcin malloc; una para crear
el array de filas y otro para reservar el espacio de
todas las celdas. Adems, sera posible utilizar fun-
ciones que trabajan con zonas de memoria para su
inicializacin de una vez (como memset()).
Punteros a funciones
El lenguaje C permite la declaracin de punteros a
funciones. Para qu puede servir un puntero a una
funcin?. Supongamos que tenemos una funcin de
ordenacin implementada (un quicksort, por ejem-
plo). Esta funcin debera poder utilizarse para orde-
nar arrays de enteros, de caracteres o de cualquier
.8 Carlos Gonzlez Morcillo ([email protected]) ::: Miguel ngel Redondo Duque ([email protected])
tipo de datos. Podamos implementar diferentes
funciones que comparen elementos de cada tipo de
datos y pasarle la referencia (puntero) de la funcin
de comparacin que corresponda a la funcin quick-
sort. As, slo tendremos una versin del algoritmo de
ordenacin, exactamente igual para todos los tipos de
datos.

01 #include <stdio.h>

02 int escribeChar (void *c, void *n)
03 {
04 int i;
05 int num = *((int *)n);
06 char *cadena = (char *)c;

07 for (i=0; i<num; i++)
08 printf ("%c", cadena[i]);

09 return 1;
10 }

11 int escribeInt (void *v, void *n)
12 {
13 int i;
14 int num = *((int *)n);
15 int *entero = (int *)v;

16 for (i=0; i<num; i++)
17 printf ("%d", entero[i]);

18 return 0;
19 }

20 void escribe (void *p, int n,
int (*funcptr)(void *, void *))
21 {
22 int tipo;
23 tipo = funcptr(p, &n);
24 tipo ? printf (" Cadena\n")
: printf (" Entero\n");
25 }

26 int main(void)
27 {
28 char cad[3] = {'a','b','c'};
29 int vect[5] = {2,3,4,5,6};

30 escribe (cad, 3, escribeChar);
31 escribe (vect, 5, escribeInt);
32 return 0;
33 }
Listado 15
En el ejemplo del Listado 15 se ha realizado un pro-
grama muy sencillo que tiene una funcin llamada
escribe que permite procesar cualquier tipo de
datos. Esta funcin se encargar de imprimir en
pantalla el tipo de datos que se le pase por argumen-
to. Para ello, escribe (lnea 20) espera como
argumentos un puntero (que apuntar a los datos
que queremos escribir en pantalla), un entero con el
nmero de elementos a escribir y una funcin con dos
argumentos de tipo puntero.

Este ltimo parmetro debe definirse con cuidado.
Estamos indicando que funcptr es un puntero a una
funcin que tiene dos argumentos void * y que
devuelve un entero. Los parntesis son totalmente
necesarios, sin ellos tendramos: int * funcptr
(const void *, const void *) con lo que decimos
que funcptr es una funcin que devuelve un puntero
a entero, lo cual es muy diferente.

En los argumentos de la funcin se han utilizado
punteros genricos void *. Cualquier puntero puede
ser forzado (mediante un cast) a void * y devuelto
de nuevo a su tipo original sin prdida de informa-
cin. De este modo, podemos pasar datos a la fun-
cin escribe sin tener que indicar previamente su
tipo.

Analizando el programa en detalle, vemos que en las
lneas 28 y 29 se declaran dos arrays de datos, uno
de enteros y otro de caracteres. Estos sern los datos
que vamos a imprimir en pantalla mediante la llama-
da a escribe. Seguidamente, en las lneas 30 y 31 se
llama a la funcin escribe. Si recordamos el prototi-
po de llamada a la funcin (lnea 20), necesitamos
pasarle un puntero con los datos, un entero y un
puntero a funcin. El cast a void * se realiza auto-
mticamente, por lo que basta con pasarle cad y
vect en cada llamada. Los punteros a funcin se
pasan con el nombre de la funcin que ha sido defini-
da previamente (escribeInt en la lnea 11 y escri-
beChar en la lnea 2).

Cuando entramos en escribeInt, lo primero que
hacemos (en la lnea 14) es recuperar el nmero de
elementos a mostrar, num. Para ello, accedemos al
contenido de la memoria apuntada por n, que forza-
mos mediante un cast a ser puntero a entero. Des-
pus, forzamos el tipo de datos del v a array de
enteros. Hecho esto, recorremos el array y escribimos
en pantalla cada elemento. De forma similar se traba-
ja en la funcin escribeChar.

Finalmente, devolvemos un entero a la funcin es-
cribe con informacin sobre el tipo de datos que se
ha tratado. Esta funcin es totalmente genrica, en la
lnea 23 llama con los argumentos que ha recibido
como parmetro. Podramos implementar una nueva
funcin que mostrara en pantalla matrices bidimen-
sionales y no habra que aadir nada a la funcin
escribe. Bastara con implementar esta nueva fun-
cionalidad y llamar a la funcin escribe desde main
pasndole como tercer argumento el nombre de la
nueva funcin.

El mecanismo de punteros a funciones nos permite
construir programas fcilmente ampliables, con
capacidad de trabajar con tipos de datos genricos en
las funciones principales del programa. As, el ncleo
principal de la aplicacin se realizar una vez y podr
trabajar con diferentes tipos de datos sin tener que
duplicar cdigo.
Y ahora qu?
Simplemente te queda practicar, practicar y practicar.
La nica forma de aprender los entresijos de un
lenguaje tan potente como C es programando con l.
Puedes profundizar en el estudio del lenguaje C y, en
concreto, en el estudio de punteros, leyendo algunos
libros y tutoriales como los que os comentamos a
continuacin. Mucha suerte y a disfrutar con uno de
los lenguajes de programacin ms usado del mundo.

Kernighan B. W., Ritchie D. M. El lenguaje de
programacin C. Ed. Prentice Hall, 1991.
Gibson T. A. Tutorial: Pointers in C and C++.
Hefty J. W. Punteros en C/C++.
Jensen T. A tutorial on Pointers and Arrays in
C.

También podría gustarte