Thursday, August 2, 2007

pointers

Section 4. Pointers

4.2: I'm trying to declare a pointer and allocate some space for it,
but it's not working. What's wrong with this code?

char *p;
*p = malloc(10);

A: The pointer you declared is p, not *p. To make a pointer point
somewhere, you just use the name of the pointer:

p = malloc(10);

It's when you're manipulating the pointed-to memory that you use
* as an indirection operator:

*p = 'H';

See also questions 1.21, 7.1, and 8.3.

References: CT&P Sec. 3.1 p. 28.

4.3: Does *p++ increment p, or what it points to?

A: Unary operators like *, ++, and -- all associate (group) from
right to left. Therefore, *p++ increments p (and returns the
value pointed to by p before the increment). To increment the
value pointed to by p, use (*p)++ (or perhaps ++*p, if the order
of the side effect doesn't matter).

References: K&R1 Sec. 5.1 p. 91; K&R2 Sec. 5.1 p. 95; ANSI
Sec. 3.3.2, Sec. 3.3.3; ISO Sec. 6.3.2, Sec. 6.3.3; H&S
Sec. 7.4.4 pp. 192-3, Sec. 7.5 p. 193, Secs. 7.5.7,7.5.8 pp. 199-
200.

4.5: I have a char * pointer that happens to point to some ints, and
I want to step it over them. Why doesn't

((int *)p)++;

work?

A: In C, a cast operator does not mean "pretend these bits have a
different type, and treat them accordingly"; it is a conversion
operator, and by definition it yields an rvalue, which cannot be
assigned to, or incremented with ++. (It is an anomaly in pcc-
derived compilers, and an extension in gcc, that expressions
such as the above are ever accepted.) Say what you mean: use

p = (char *)((int *)p + 1);

or (since p is a char *) simply

p += sizeof(int);

Whenever possible, you should choose appropriate pointer types
in the first place, instead of trying to treat one type as
another.

References: K&R2 Sec. A7.5 p. 205; ANSI Sec. 3.3.4 (esp.
footnote 14); ISO Sec. 6.3.4; Rationale Sec. 3.3.2.4; H&S
Sec. 7.1 pp. 179-80.

4.8: I have a function which accepts, and is supposed to initialize,
a pointer:

void f(ip)
int *ip;
{
static int dummy = 5;
ip = &dummy;
}

But when I call it like this:

int *ip;
f(ip);

the pointer in the caller remains unchanged.

A: Are you sure the function initialized what you thought it did?
Remember that arguments in C are passed by value. The called
function altered only the passed copy of the pointer. You'll
either want to pass the address of the pointer (the function
will end up accepting a pointer-to-a-pointer), or have the
function return the pointer.

See also questions 4.9 and 4.11.


4.9: Can I use a void ** pointer to pass a generic pointer to a
function by reference?

A: Not portably. There is no generic pointer-to-pointer type in C.
void * acts as a generic pointer only because conversions are
applied automatically when other pointer types are assigned to
and from void *'s; these conversions cannot be performed (the
correct underlying pointer type is not known) if an attempt is
made to indirect upon a void ** value which points at something
other than a void *.

4.10: I have a function

extern int f(int *);

which accepts a pointer to an int. How can I pass a constant by
reference? A call like

f(&5);

doesn't seem to work.

A: You can't do this directly. You will have to declare a
temporary variable, and then pass its address to the function:

int five = 5;
f(&five);

See also questions 2.10, 4.8, and 20.1.

4.11: Does C even have "pass by reference"?

A: Not really. Strictly speaking, C always uses pass by value.
You can simulate pass by reference yourself, by defining
functions which accept pointers and then using the & operator
when calling, and the compiler will essentially simulate it for
you when you pass an array to a function (by passing a pointer
instead, see question 6.4 et al.), but C has nothing truly
equivalent to formal pass by reference or C++ reference
parameters. (However, function-like preprocessor macros do
provide a form of "call by name".)

See also questions 4.8 and 20.1.

References: K&R1 Sec. 1.8 pp. 24-5, Sec. 5.2 pp. 91-3; K&R2
Sec. 1.8 pp. 27-8, Sec. 5.2 pp. 91-3; ANSI Sec. 3.3.2.2, esp.
footnote 39; ISO Sec. 6.3.2.2; H&S Sec. 9.5 pp. 273-4.

4.12: I've seen different methods used for calling functions via
pointers. What's the story?

A: Originally, a pointer to a function had to be "turned into" a
"real" function, with the * operator (and an extra pair of
parentheses, to keep the precedence straight), before calling:

int r, func(), (*fp)() = func;
r = (*fp)();


It can also be argued that functions are always called via
pointers, and that "real" function names always decay implicitly
into pointers (in expressions, as they do in initializations;
see question 1.34). This reasoning, made widespread through pcc
and adopted in the ANSI standard, means that


r = fp();

is legal and works correctly, whether fp is the name of a
function or a pointer to one. (The usage has always been
unambiguous; there is nothing you ever could have done with a
function pointer followed by an argument list except call the
function pointed to.) An explicit * is still allowed (and
recommended, if portability to older compilers is important).

See also question 1.34.

References: K&R1 Sec. 5.12 p. 116; K&R2 Sec. 5.11 p. 120; ANSI
Sec. 3.3.2.2; ISO Sec. 6.3.2.2; Rationale Sec. 3.3.2.2; H&S
Sec. 5.8 p. 147, Sec. 7.4.3 p. 190.

No comments: