Was trying to brush up my C concepts and now it seems all jumbled up :( in pointers, of course
This way of assigning arr to ptr works since we say arr[] will decay to *arr
int arr[] = {1, 2, 3};
int *ptr = arr; // Access arr with ptr
But directly assigning the array to *ptr doesn't work
int *ptr = {1, 2, 3};
printf("%d\n", ptr[0]); // Segmentation fault
My understanding is that int arr[] = {} has a special meaning where a contiguous chunk of stack space is allocated and is directly referred to by the name arr
Trying to do the same thing with int *ptr = {} is just confusing the compiler ??
CodePudding user response:
The variable ptr is a scalar object. You may initialize a scalar object with a braced list that contains only one expression.
From the C Standard (6.7.9 Initialization)
11 The initializer for a scalar shall be a single expression, optionally enclosed in braces. The initial value of the object is that of the expression (after conversion); the same type constraints and conversions as for simple assignment apply, taking the type of the scalar to be the unqualified version of its declared type.
So instead of:
int *ptr = {1, 2, 3};
you may write for example:
int *ptr = { 3 };
that is equivalent to:
int *ptr = 3;
and does not make a sense and the compiler can issue a message that an integer is converted to a pointer without casting.
Instead of the braced list you could initialize the pointer with a compound literal like:
int *ptr = ( int [] ){1, 2, 3};
In this declaration there is created an unnamed array of the type int[3] and the address of the first element of the array is assigned to the pointer.
CodePudding user response:
{1, 2, 3} does not mean “an array”. In a declaration, it is a list of initial values for some object with multiple parts.
In int arr[] = {1, 2, 3};, {1, 2, 3} is a list of three values to use to initialize the array arr.
In int *ptr = {1, 2, 3};, {1, 2, 3} would be a list of three values to use to initialize the pointer ptr. But ptr does not have multiple parts. All it has is one value, a memory address. So {1, 2, 3} would provide 1 to initialize the address, which is a problem because 1 is an int, not an address, so the compiler should issue a diagnostic message for that. And there is nothing for 2 or 3 to initialize, so the compiler should issue a diagnostic message for that.
You can use a compound literal to create an unnamed array directly in source code. A compound literal has the form (type) { initial values }. So you can create an array of int with (int []) {1, 2, 3}.
You can declare a pointer and initialize it to point to an array by using a compound literal as the initial value:
int *ptr = (int []) {1, 2, 3};
(Note that the array is automatically converted to a pointer to its first element, so ptr is initialized to that address.)
