Lab 6 --
Introduction To Pointers
CPE101
Winter 2008
Updates and Corrections
Due Date:
By the end of your last lab of week 6.
Objectives
Overview
In
this lab you will be exploring the concept of a pointer. When you
declare a variable, space is set aside for it in memory (where in
memory is outside the scope of this class). Each memory location
in the computer has an address. You can think of that address
like a mailbox number.
Thus far in this class we have referred to variables only by their
names. However, we can
also refer to variables by their address. A pointer is simply a
variable that holds the address of another variable. In essence,
the pointer variable "points" to where the second variable is located
in memory.
Why in the world would we want a
variable that points to the location of another variable?
Excellent question. In this lab we will explore the concepts of
variables having an address, of a variable pointing to another
variable, and finally why we would need to use a pointer in the first
place.
The first part of this lab will be a
step-by-step exploration of the concepts necessary to understand
pointers. I ask that you do not read ahead, but instead complete
the tasks asked of you as we go. The second part of the lab will
contain programming tasks for you to complete and hand in.
Pointer
Exploration
What's
an address?
Let's begin with a sample program to
explore the concept of a variable being stored in a particular memory
location. When you refer to a variable by its name you get the
value of the variable. For example, the following code prints the
value of a variable called x:
printf("%d", x);
However, if you want to know the address of a variable, you put the '&' (ampersand) symbol in front of it. The following code prints the address of a variable called x:
printf("%d", &x);
Now consider the following code:
#include <stdio.h>
int
main(void)
{
int x, y;
double z;
x = 5;
y = -69;
z = 1.0;
printf("x is %d. The address of x is %u.\n", x, &x);
printf("y is %d. The address of y is %u.\n", y, &y);
printf("z is %f. The address of z is %u.\n", z, &z);
return 0;
}
The code declares 3 variables. It
then prints the value and the address of each variable. Compile this code using gcc, but without all the usual flags.
gcc
addresses.c
What were the addresses of the variables? Which variable had the
smallest address? How many bytes does the variable z take
up? How many bytes does
the variable y take up?
What's
a pointer?
Now let's declare and use our first pointer. To declare a pointer
you put an asterisk '*' in front of the name of the variable. The
following code declares a pointer to an integer:
int *x;
This variable is now set up to hold the address of some other
integer. Copy, compile, and run the following code:
#include
<stdio.h>
int main(void)
{
int x;
int *px;
x = 5;
px = &x; /* set px to the address of x */
printf("x is %d. The address of x is %u.\n", x, &x);
printf("px is %u. The address of px is %u.\n", px, &px);
return 0;
}
Notice that the value of px (the pointer) is the same as the address of
x. In essence, px knows where x is stored in memory. Notice
that px has an address, too. It is a variable stored in memory
just like any other variable.
Now you should be asking yourself, so what? Can I do anything
with this pointer? We can use px to get the value of x, even
though px only stores the address of x. (It's similar to being
able to go get someone's mail if you know their address.) We can
get the value of x by "dereferencing" px. When you dereference a
pointer, you are telling the machine to go to the address the pointer
is pointing at and get that value out. So, if px is pointing at
x, we get the value of x when we dereference px.
Let's look at an example. To dereference a pointer we put an
asterisk in front of it. (Note that this is completely different
from using an asterisk to declare a pointer.) Add the following
printf statement to the code above:
printf("Value
of *px is %d.\n", *px);
Now compile and run the code. The value of *px (px dereferenced)
is the same as x.
We can modify x the same way, if we dereference px and set it to
something, we change x. Add this code to your file:
*px =
6;
printf("x is
now %d\n", x);
We are telling the computer to go to where px is pointing and change it
to 6. Compile and run the code to make sure it works as
advertised.
The big
deal...
We are finally ready to see an example of how this is useful.
Consider the following main and swap functions. The swap function
takes two integers as parameters and is supposed to swap their
values. Do
not compile and run the code yet.
#include
<stdio.h>
void swap(int
a, int b);
int main(void)
{
int x, y;
x = 5;
y = 10;
swap(x,y);
printf("x = %d, y = %d\n", x, y);
return 0;
}
void swap(int
a, int b)
{
int temp;
temp = a;
a = b;
b = temp;
}
What do you think will be printed to the screen? What will the
values of x and y be? Now compile and run the code. Why
didn't x and y get swapped? The reason is that we passed the values of x and y to a and b.
We initialized a with the value of x and b with the value of y.
Then we swapped a and b. The variables x and y never got
touched. The swap() function has no way to access x and y.
The following code modifies the swap function so we pass the addresses of x and y. That
way the swap function knows where in memory x and y are located.
Since swap knows where they are, it can change them. Copy,
compile, and run the following code:
#include
<stdio.h>
void swap(int
*a, int *b);
int main(void)
{
int x, y;
x = 5;
y = 10;
swap(&x,&y); /* pass the addresses of x and y */
printf("x = %d, y = %d\n", x, y);
return 0;
}
void swap(int
*a, int *b)
{
int temp;
temp = *a; /* set temp to the value a is pointing at */
*a = *b; /* go to where a is pointing and give it
* the value of what b is pointing at */
*b = temp; /* go to where b is pointing and set it to temp */
}
Notice in main that we pass the
addresses of x and y, instead of their values. Now a holds the
address of x and b holds the address of y. First we set temp
equal to the value a is pointing at (5). Then we set the value of
where a is pointing (x) equal to the value of what b is pointing at
(10). Finally, we set the value of where b is pointing to the
value of temp (5).
The big point of all this is that by knowing the addresses of x and y,
swap() was able to modify those variables. It had no direct
knowledge of x and y, but only knew where in memory they were
located. Until know, we've only been able to pass the values of
variables to functions and get a single return value. Now we can
actually modify variables inside of functions.
You should now be able to answer the question, "Why do we use '&'
with scanf and not with printf???" Printf only needs the value of
whatever variable you want print. Scanf needs to modify the
variable you pass to it. It scans something in from the keyboard
and then stores it in memory to the address of the variable you pass to
it. A function can not
modify a variable if you only pass its value.
Output
Parameters
A function paramter
that is a pointer can be referred to as an output parameter. The
function is relaying information to you via that parameter.
Consider a function that has two parameters. The first is an
integer test score. The second is a pointer to a char that the
function will fill in with a letter grade based on the test
score. The return value of the function will be an integer: 0 if
the test score was valid (0-100) or -1 if the test score was
invalid. The function prototype might look like this:
The first parameter, score, is an input parameter. It is
information the function needs to do its job. The second
parameter, grade, is an output paramter. The function fills it in
with information that we need.
Can I declare a pointer to any kind of
variable?
Yes. You can have a pointer to an int, double, char, another
pointer, or even structures (we'll cover these soon). Passing a variable's address to a
function allows the function to modify the variable. However,
there is another reason we might want to pass a variable's address
instead of its value. Consider the following structure:
typedef
struct
{
char make[20];
char model[20];
int numDoors;
double mpg;
} Car;
How much space in memory does every variable of type Car take up?
Can you guess? Here is a simple program to find out. It
uses the sizeof() operator. Although it looks like a function
call, sizeof() is an operator just like = or +. To use it, simply
put the thing you want to know the size of in the parentheses. It
will evaluate to the number of bytes in the thing you're asking
about. You may put a variable or a type in the parentheses.
Copy, compile, and run the following code:
typedef
struct
{
char make[20];
char model[20];
int numDoors;
double mpg;
} Car;
int main(void)
{
Car car;
printf("Number of bytes in Car: %d\n", sizeof(car));
return 0;
}
Did you guess correctly? When we
pass a Car to a function all this information must be copied to the
function parameter. If a structure is 100 bytes, then 100 bytes
of information get passed to the function. Where these bytes get
put is outside the scope of this class. However, we want you to
know that it is sometimes undesirable to pass that much information in
a function call. A pointer only takes up 4 bytes. So, if we
pass a pointer to a Car, we only take up 4 bytes in memory, but still
have access to all the information about the structure.
Here is code that passes the value of a
structure to a function:
/*
structure definition not shown */
void
printCar(Car c);
int main(void)
{
Car car = {"Volvo", "V70", 5, 24.0};
printCar(car);
return 0;
}
void
printCar(Car c)
{
printf("Size of 'c': %d\n", sizeof(c));
printf("Make: %s\n", c.make);
printf("Model: %s\n", c.model);
printf("Doors: %d\n", c.numDoors);
printf("MPG: %.2f\n", c.mpg);
}
Copy, compile, and run the code.
Notice the size of c is 52 bytes. Now let's modify the code to
work with a pointer to a Car. First you need to know that we
cannot use the '.' operator with a pointer to a structure. A
pointer to a Car
doesn't have a model or make. It is simply an address. So,
we need to dereference the pointer before we can access those
attributes. Assume we have a pointer to a Car called 'c'.
We could dereference it by writing (*c).make. But, that is ugly
and cumbersome, not to mention it's hard to remember where the
parentheses go. So, we can instead us the -> operator.
To dereference a pointer to a structure and access one of its fields we
can type c->make. Here is the code modified to work with
pointers:
/*
structure definition not shown */
void printCar(Car
*c);
int main(void)
{
Car car = {"Volvo", "V70", 5, 24.0};
printCar(&car);
return 0;
}
void
printCar(Car *c)
{
printf("Size of 'c': %d\n", sizeof(c));
printf("Make: %s\n", c->make);
printf("Model: %s\n", c->model);
printf("Doors: %d\n", c->numDoors);
printf("MPG: %.2f\n", c->mpg);
}
Copy, compile, and run this code.
This time, how big was the variable c? We saved a lot of memory
by passing the address of the structure instead of the entire
structure. Furthermore, we could now modify the car structure
inside the function.
Lab Requirements (Or the answer to "So what am I supposed to hand in?")
Handing in
Your Source
Electronically . . .
11:59
vogon ~$ handin mhaungs Lab06-yy part1.c part2.c part3.c
Extra Challenges (What do I do if I finish the lab but want more
practice?)