"A. Bolmarcich" <aggedor@earl-grey.cloud9.net> wrote in message
On 2011-06-07, Paul <pchristor@yahoo.co.uk> wrote:
"A. Bolmarcich" <aggedor@earl-grey.cloud9.net> wrote in message
news:slrniuqah1.16ra.aggedor@earl-grey.cloud9.net...
On 2011-06-04, Paul <pchristor@yahoo.co.uk> wrote:
"A. Bolmarcich" <aggedor@earl-grey.cloud9.net> wrote in message
news:slrniukoa8.1fj5.aggedor@earl-grey.cloud9.net...
On 2011-06-03, Paul <pchristor@yahoo.co.uk> wrote:
[snip]
The C++ sees it as: "An object of array type contains a
contiguously allocated nonempty set of N sub-objects of type T."
Pointing to an array is pointing to an object of array type.
Pointing to a sub-object in an array is pointing to something other
than an object of the array type of the array that the sub-object is
in.
You have quoted some text from that seems to describe an object of
array
type.
As I said , and you have proved me correct, you can only see an array
as
"a
single array type object". You cannot see it at as array of integer
objects( or whatever type it is).
I have quoted the C++ standard. Seeing it as the C++ standard does
is the proper way to see it when discussing C++, such as in the
comp.lang.c++ newsgroup. Seeing it another way is seeing it in a
way different from the C++ standard.
This is not a definition of the term "array", this piece of text you have
quoted is not the C++ std's way of saying an array is definitely not an
sequence of objects, this is saying and array TYPE object is ....and it
doesn't mean this is the only possible way to consider an array.
Because an array is an object of array type, and not of any other
type, what the C++ standard states about an object of array type is
about an array.
What I quoted says that an array is a sequence of objects. The
phrase used in the standard is "a contiguously allocated set of
sub-objects". Here is part of a standard definition of the word
contiguous (the word from which the adverb contituously is derived):
"next or together in sequence" (from
http://oxforddictionaries.com/definition/contiguous). The
sub-objects an (object of type) array contains are together in a
sequence.
The C++ standard considers an array to be a sequence of sub-objects.
That is what an array should be considered to be when discussing C++.
With the following code..
int arr[7];
int* p1 = arr;
int (*p2)[7] = &arr;
Both the pointers point-to the first element in the array, and both
pointers
also point-to the array of integer objects. They are just different
types
of
pointers.
They both point to exactly the same location.
Both pointers do not point to the first element of the array. p1
points to the first element of the array; p2 points to the array.
An array is not the same as its first element.
The address of an array is the same as the address of the first
element,
and
a pointer is this address.
The first element of an array is at the same memory location as the
array. The value of a C++ pointer is not only a memory location; it
is a typed memory location.
Rubbish the value of a pointer object does not contain any typeinfo. Its
a
simple integer value that represents the address of the pointed to
object.
The typeinfo is associated with the type of the pointer. In C++ a
pointer is not a simple integer value; it a typed value.
[Continuing with what I wrote in a single paragraph that you split.]
Ignoring their types, the p1 and p2 have
the same value; however because p1 and p2 have different types, they
do not point to the same entity. p1 points to the first element of
the array, not to the array; p2 points to the array.
They point to exactly the same location.
The type of a pointer does not define what is pointed to.
In C++ the type of the pointer determines the type of the result of
dereferencing the pointer. Here, again, is what the C++ standard
states about the unary * operator: "If the type of the expression is
pointer to T, the type of the result is T."
If the type of the expression is a pointer is to an array of T, the
result of dereferencing its is type array of T. An attempt to access
the result of dereferencing a pointer to array of T when the value of
the pointer is not the address of an array of T object is undefined
behavior (see paragraph 15 of 3.10 (Lvalues and rvalues) of the C++
standard).
These posts are long enough without having to mention undefined
behavior when a value is not valid.
They both point to exactly the same place and they both point to
exactly
the
same array. Whether you view it as pointing to an array of pointing to
the
first element is just a POV.
They do not both point to exactly the same array. One points to an
array; the other may point to an element of that array. The point of
view of the C++ standard is that a
int (*)[]
points to an array of int while a
int *
points to an int, the int it points to may be an element in an array
of int.
This is completely made up. Where does the standard define that a pointer
type defines what is pointed to?
In fact where does the C++ standard define the term "points to"? It
doesn't.
One place the C++ standard uses the term "points to" is paragraph 2 of
4.10 (Pointer conversions):
An rvalue of type "pointer to cv T," where T is an object type, can
be converted to an rvalue of type "pointer to cv void." The result
of converting a "pointer to cv T" to a "pointer to cv void" points
to the start of the storage location where the object of type T
resides, as if the object is a most derived object (1.8) of type T
(that is, not a base class subobject)."
"Points to" the start of a storage location is what a value of a
pointer type does. A pointer to int points to an int, accessing what
it points to when its value is not the address of an int is undefined
behavior (see paragraph 15 of 3.10 (Lvalues and rvalues) of the C++
standard.)
[snip]
The declarations were
int arr[7];
int* p1 = arr;
int (*p2)[7] = &arr;
p1 is a pointer to an integer.
No its a pointer to 7 integers.
The type of p1 is int*; it is a pointer to int. The values of p1 and
p2 are the same storage location. Because p1 and p2 havd different
types, they point to different entties. p1 to and int and p2 to an
array of 7 int.
Saying that a pointer is to whatever is at the storage starting at
the value of the pointer ignores the fact that C++ pointer are typed.
Ignoring the pointer type would mean that for the declaration
int arr[3][5], *p;
setting the value of p to the address of the first integer in the
2-dimensional array would mean that p is a pointer to 1, 5, and 15
integers at the same time. It is a pointer to 15 integers because
15 were allocated starting at the storage location that p has as
its value. It is a pointer to 5 integers because the first of 3
arrays of 5 integers were allocated starting at the same storage
location. It is a pointer to 1 integer because the first integer
in the first of 3 arrays of 5 integers were allocated starting at
the same storage location.
For the declaration
int arr[3][5][7][9][11], *p;
setting the value of p to the address of the first integer in the
5-dimensional array would mean that p is a pointer to 1, 11, 99,
693, 3465, and 10395 integers at the same time.
A pointer to a member of a POD (plain old data) struct is a similar
situation. Ignoring its type, with
struct s_t {
int a, b, c, d, e;
} s;
int* pi = &s.a;
pi points to 5 integers.
With
struct s_t {
int a, b;
double c, d
long e;
} s;
int* pi = &s.a;
pi points to 2 integers, 2 doubles, and a long. What versitility
an int* has in a view that ignores the pointer type!
Because pointers are typed in C++, an int* points to an int. Just
because a pointer to int is treated as if it is to an element of an
array of int when an integral value is added to the pointer does not
mean that it does not point to an int.
This is similar to just because array-to-pointer conversion is
implicitly done in some contexts that an array is the same as a
pointer.
What C++ has available as a pointer to
multiple integers is a pointer to an array of integers, such as p2.
p1 is a pointer to an integer, accessing what p1 points to access
that integer.
Accessing what p2 points to also accesses the same integer.
They are just different types of pointer but they both point to exactly
the
same thing.
p1 and p2 do not point to the same thing. p1 points to an int, p2
points to an array of int. If p1 an p2 pointed to the same thing,
it would not be necessary to dereference p2 twice to get the same
int that is gotten by dereferencing p1 once.
[snip]
An element in an array has a different type than the array the
element is in. The type of p1 is int*. When p1 points to an element
of an array of int, it is pointing to an an int, not to an array.
The type of a pointer to an array of int is of the form int(*)[].
The C++ standard is clear about this in the following example from
8.1 (Types).
int // int i
int * // int *pi
int *[3] // int *p[3]
int (*)[3] // int (*p3i)[3]
int *() // int *f()
int (*)(double) // int (*pf)(double)
name respectively the types "int," "pointer to int," "array of 3
pointers to int," "pointer to array of 3 int," "function of (no
parameters) returning pointer to int," and "pointer to a function
of (double) returning int.
That is the TYPE of pointer, nobody is disputing the pointer TYPE. But a
pointer TYPE does not define what is pointed to by that pointer.
Writing, as you did above that "No its a pointer to 7 integers." as a
response to "p1 is a pointer to an integer.", where p1 is to type
int*, is disputing that in C++ the result of dereferencing a pointer to
T is a T, as opposed to many Ts. A pointer type defines what may be
validly pointed to. Dereferencing a pointer to T whose value is not
an address of T is undefined behavior.
A pointers' type simply tells us that that pointer is capable of
addressing
objects of the given type.
For example:
char* cp;
This pointer is capable of addressing char objects, it doesn't point to a
char object.
The value cp may validly be the address of a char object, the address
of a char object +1, or NULL. Dereferencing cp when its value is not
the address of a char object is undefined behavior. These posts are
long enough without having to mention undefined behavior when a value
is not valid.
The distinction between a pointer to an array and a pointer to an
element of that array is important for multidimensional arrays as
arrays of arrays in C++. The declaration
int x[3][5];
declares an array of 3 array of 5 int. The expression x[i][j] is
equivalent to *(*(x+i)+j). When that expression is evaluated
This is not what the Std says. The standard says that each subsctipt will
be
calucalated individually.
i has a multiplier of 5xsizeof(int), j has the multiplier of
1xsizeof(int).
Where is what follows are the subscripts not calculated individually?
- x is an array of 3 array of 5 int
- array-to-pointer conversion results in a pointer to the first array
of 5 int in x
- adding i results in a pointer to the i-th array of 5 int in x
- dereferencing that results in the i-th array of 5 int in x
- array-to-pointer conversion results in a pointer to the first int in
the i-th array of 5 int in x
- adding j results in a pointer to the j-th int in the i-th array of
5 int in x
- dereferencing that result in the j-th int in the i-th array of 5 int
in x
Multidimensional arrays in C++ depend on the distinction between a
pointer to an array and a pointer to an element of that array. Not
understanding this distinction may lead to confusion.
Unlike p1, p2 is a pointer to an array type object. p2 points to an
array. p1 does not point to an array; it points to the first
element of the array that p2 points to.
p1 and p2 point to exactly the same place, and the same array.
p1 and p2 have different types. p1 points to an int, not to an
array. p2 points to an array of 7 int.
They have different types, but they both point to the same location.
In C++ pointers are typed; they are not only a storage location value.
The type of the pointer determines type used when accessing what the
pointer points to.
Consider a stack frame pointer, pointing to the stack. It doesn't
actually
point the whole stack at the same time but nonetheless its still
considered
to be pointing to the stack.
There is a significant difference between a stack frame pointer and a
C++ pointer: a C++ pointer is typed. In C++, the result of
dereferencing a pointer to T, where T is a type, is a T.
No, dereferencing a pointer of type pointer-to-T, will result in a T
if
it
points to a valid T object.
The pointer type declares a pointer object that can address an object
of
the
decalred type , for example:
int *p = 0;
p = &ix; /*assume x is valid int*/
p = new int[64];
What I have written is about what a C++ compiler does, along the
lines of what the C++ standard does in the folliwng sentence about
the unary * operator: "If the type of the expression is pointer to T,
the type of the result is T." These post are long enough without
adding possible undefined behaviors when the program runs due to the
value of a pointer to a non-polymorphic type.
This is explaining just what I said, if the type of the pointer is T then
derefernecing it addresses a T.
Assuming it points to a valid T type object.
This is not defining what p points to. p points to a location and
whatever
is stored there is what is pointed to.
Whatever is at the storage location that is the value of a pointer to
T is treated as being of type T. The type of the pointer defines how
what is pointed to is treated. If what is at the storage location is
not an object of the appropriate type, the behavior is undefined.
These posts are long enough without having to mention undefined
behavior when a value is not valid.
Here the pointer p is declared to be a pointer of type ptr-to-int, it
means
this pointer must be capable of pointing to a valid int object., It has
nothing to do with what it points to.
The above the pointer is shown pointing to nothing, an integer and an
array
of integers.
What is shown is p having the null pointer value, pointing to an
integer, and pointing to the first integer in an array of integers
(the result of the expression (new int[64]) is a pointer to int,
not a pointer to array of int).
No what is returned from the new expression is a pointer to the first
element *of an array* of integers.
When the pointer is used as a pointer to the array like so:
p[3];
The expression is evaluated as:
*(p+ 3xsizeof(int))
Where p is a pointer to the beginning of the array , and the offset to
the
individual element is added.
The pointer to the first element of the array is a pointer to int (type
int*). The expression
(p + 3xsizeof(int))
is a pointer to the 3xsizeof(int) element relative to the array element
that p is treated as pointing to. If p points to the first element of
and array and sizeof(int) is 4, then
(p + 3xsizeof(int))
points to the 13th element of the array (first plus 12).
Pointers in C++ are typed. The storage location that is pointed to
when adding a pointer and an integral value depends on the pointer
type.
This pointer is used as a pointer to the array, not to only just one
element. And it can be used as a pointer to the array because it points
to
an array.
In the example, the pointer is not used as a pointer to an array of
int. p is a pointer to int. It may point to an element in an array
of int. A pointer to an element in an array of int is not same as a
pointer to the array. The former is of type int*; the latter is of
type like int(*)[64]. If p could be used as a pointer to an array (of
64 int), the following would be allowed, but it isn't.
int (*parr)[64] = p;
Yes its TYPE is int*, but this does not mean it can only point to one
single
integer. Its type does not define what it points to.
The type of a pointer defines what may be validly pointed to. An int*
may validly point to one single integer.
Now consider the following code:
int *p;
What does this point to? Uninitialised memory that could contain
anything.
The pointer type only tells us that the declared pointer object is
capable
of pointing to an integer, it does not define what is pointed to, as
you
have tried to imply.
It is undefined what p points to. I have used shorter phrases, such
as "is a T", rather than a longer ones, such as "is of type T". You
are misreading "is a T" as "is an object of type T that exists when
the program runs".
Well you have said that, becuase it is of type int*, it can only point to
an
int.
Now you are saying it is undefined what it points to.
I repeat..the pointer type does not define what is pointed to.
I can repeat, too. As I wrote above, where cp is of type char*
The value cp may validly be the address of a char object, the address
of a char object +1, or NULL. Dereferencing cp when its value is not
the address of a char object is undefined behavior. These posts are
long enough without have to mention undefined behavior when a value
is not valid.
You are shortening the term a "pointer type ptr-to-T" to mean the same as
a
"pointer to T". But the two terms do not mean the same thing.
One term is specifically describing the pointer type. The other is
referring
to whatever is at the location pointed to.
I have not used the term "ptr-to-T". Where is it in the C++ standard?
The value of a pointer to T, where T is a non-polymorphic type, may
validly be the address of an object of type T, the address of an object
of type T +1, or NULL. Dereferencing a pointer to T when its value
is not the address of an object of type T is undefined behavior. These
posts are long enough without having to mention undefined behavior when
a value is not valid.
The result of using a stack frame pointer as an address operand
depends on what the operator does. Different operators move 1, 2, 4,
8, or some other number of bytes to or from storage starting at the
address given by the address operand.
What you seem to be talking about here is what I know as the PTR
keyword
in
assembly, it may change in different assemblers.
I don't really see what this has to do with the fundamental concept
that
a
SFP "points to" a stack frame.
In that case, you don't really see the difference between a value
with the correct number of bits to be used as an address in assembler
and a C++ pointer. C++ pointers are typed.
The term "pointer" is not redefined because C++ pointers are typed.
A pointer is defined in the C standard and C++ inherits this defintion. A
pointer of type char* can point to a single char, and array of chars or
nothing at all. This is the view of the C standard and also Bjarne
Stroustrup in one of his C++ books.
When I run a C compiler on the statement that tries to set a char*
to point to an array of char, that is, set the value of a pointer
to char to the address of an array of char, as in
char arr[4], *pc = &arr;
I get a diagnostic message. Setting a char* to point to an element
in an array of char, that is, set the value of a pointer to char to
the address of an element of an array of char, as in
char arr[4], *pc = &arr[0];
is allowed.
In a previous post you partially quoted a glossary entry for char*
by Bjarne Stroustrup. What you quoted omitted: "Typically assumed
to point to a C-style string. Prefer a standard library string over
a C-style string when you can. TC++PL 2.3.3, 13.5.2."
The context of the glossary entry is C-style strings. The glossary
did not contain similar entries for other specific types, such as
int*, or an entry for T* that states a T* is a pointer to a T or to
an array of T.
Allowing
char v[] = "Annemarie";
is a special case for arrays of char and C-style strings.
When writing at a detailed level, Bjarne Stroustrup is careful
about the details, as in the following from an example on page
91 of the Special Edition of "The C++ Programming Language".
int v[] = { 1, 2, 3, 4 };
int* p1 = v; // pointer to initial element (implicit conversion)
Note that p1 is a pointer to the initial element, not a pointer to
the array.
In an eariler edition of the same book he wrote (page 53 of the
second edition):
Pointers to arrays and pointers to functions unfortunately need a
more complicated notation.
int* pi;
char** cpp; // pointer to pointer to char
int (*vp)[10]; // pointer to array of 10 ints
int (*fp)(char, char*); // pointer to function
// taking (char, char*) arguments
// and returning int
An int(*)[] points to an array of int; a int* does not, but it may
point to an element in an array of int.
With an array like so:
[int][int][int][int][int][int][int][int]
If a pointer 'p' points to the beinning of this array then the
expression:
p+1;
points to the second element.
p+2;
points to the third element, and so on.
The pointer 'p' points to an array of integers, not only one single
integer.
The TYPE of 'p' can only be int*, because if it were of type int (*)[8]
then
p+2 would point away past the array.
The pointer p points to an int. It is the semantics of the binary +
operator that p+i points to the i-th element of an array when p
points to the first element of the array. Those semantics also treat
an int(*)[8] as pointing to an element in an int[][8] array when an
integral value is added to an int(*)[8].
The pointer 'p' is used as a pointer to an array of integers, it could
not
be used as a pointer to an array if it only pointed to one single object.
It is a different concept from pointing to one single array type object.
The pointer p cannot be used as a pointer to an array of integers. If
it could, the last initialization in the following declaration would be
allowed, but it isn't.
int arr[4], (*parr)[4] = &arr, *p = parr;
The pointer p points to a single int type object. It may point to an
element in an array of int. In which case, it is pointing to a single
int in that array.
When p is pointing to a single int in an array, it due to the
semantics of adding a pointer and an integral value that (p+1) points
to the next element in the array. Pointing to an element in an array
is a different concept than pointing to an array.
The C++ standerd does not disgaree with me but it disagrees with you.
integers, or nothing at all.