Reputation: 133
I learning these concepts, please help me why the following code is throwing a segmentation fault. My intention in this code is to print capital letter D and move to next address. Please explain me. thank u.
main()
{
char *ptr="C programming";
printf(" %c \n",++*ptr);
}
Upvotes: 2
Views: 8712
Reputation: 21609
This will print D
then increment the pointer to the next char.
#include <stdio.h>
main()
{
char *ptr="C programming";
printf(" %c \n",(*ptr++) + 1);
}
You cannot increment the character value in place in memory as it is a character literal and as such most probably stored in the data section of your binary executable. it's read only and any attempts to alter it will lead to undefined behaviour (segfault probably ~100% of the time).
You should make a habit of declaring such literals const
. Indeed c++ will pull you up on this if it has any extra level of warnings enabled.
The above code works because it is not altering the memory, it is simply adding 1 to the value for the purpose of passing to printf
.
Another option is to use array syntax instead of pointer syntax, you are declaring the string to be on the stack and so it is writeable.
To do this the syntax is
char ptr[]="C programming";
You can think of a pointer just pointing anywhere, you give the compiler an option to put it wherever it likes, so it sticks it in read only memory. When you use array syntax is has to be right there, where you declare it.
Upvotes: 3
Reputation: 24910
You are trying to modify a string literal, which is a non-modifiable object. That's why you get a segmentation fault.
Even if you simply call ptr[0]++
, it will be a segmentation fault.
One solution is to change the declaration to:
char ptr[] = "C programming"
Then you can modify the char array.
They look similar? Yes, the string literal is still non-modifiable, but you have declared an array with its own space, which will be initialized by the string literal, and the array is stored in the stack, and thus modifiable.
Here is a full code example:
#include <stdio.h>
int test() {
char str[]="C programming";
char *ptr = str;
while(*ptr != '\0') {
// print original char,
printf("%c \n", *(ptr++));
// print original char plus 1, but don't change array,
// printf("%c \n", (*(ptr++))+1);
// modify char of array to plus 1 first, then print,
// printf("%c \n", ++(*(ptr++)));
}
return 0;
}
int main(int argc, char * argv[]) {
test();
return 0;
}
Tip: you should only enable one of the printf()
lines at the same time, due to ++
operator.
Note that we declared a ptr
and a str
, because we can't use ++
operation on an array (you can't change the address of an array), thus ++str
will get a compile error, while ++ptr
won't.
(To answer your comment)
You might also want to know more about pointer or address or array or Linux process memory layout_ or data sections of a C program; try searching on Google, or refer to books like The C Programming Language, 2nd Edn and The Linux Programming Interface — though this is a question about C rather than Linux. There's also The Definitive C Book Guide and List on Stack Overflow.
Upvotes: 7
Reputation: 6101
why the following code is throwing a segmentation fault
++*ptr
means *ptr = *ptr + 1
in C, string literal is a non-modifiable object. You are trying to dereference ptr
and set value for the address that ptr
points to.
Upvotes: 0
Reputation: 1742
Assuming you are trying to increment the ascii value of C
by one and print that and move the character pointed to forward, you want to use *ptr++ + 1
.
Upvotes: 0