1
+
2
+ #include <stdio.h>
3
+
4
+ /*
5
+ * Pointers Arithmetic.
6
+ * In this program, I am experimenting with pointer arithmetic to help you
7
+ * understand the relationship between arrays and pointers. The goal is to
8
+ * help you get more comfortable with pointers, memory addresses, and arrays.
9
+ * The more you understand about this important concepts, the easier it will
10
+ * be to write powerful programs and to debug errors that are going to occur
11
+ * when you make mistakes. It can be a little challenging at first, but just
12
+ * keep experimenting and reviewing these examples. It will become more clear
13
+ * as you practice.
14
+ */
15
+
16
+ int main (){
17
+
18
+ /*
19
+ * When you delcare a pointer, you are creating a special type of variable.
20
+ * Because you used the special '*' symbol in the declaration, the compiler
21
+ * treats your variable in special ways when you use math operators.
22
+ *
23
+ * First, let's understand how arrays and pointers relate. Below, I show you
24
+ * two techniques to get a pointer (an address) of the first element of an
25
+ * array.
26
+ */
27
+
28
+ int a10 [10 ] = {0 };
29
+ int * ptr1 = a10 ;
30
+ int * ptr2 = & a10 [0 ];
31
+ printf ("ptr1 = %p\nptr2 = %p (expect equal values)\n" , ptr1 , ptr2 );
32
+ printf ("\n" );
33
+
34
+ /*
35
+ * Now, let's do some math on these pointers and see what happens.
36
+ * If you look at the output of this code, you will see that adding
37
+ * 1 to the pointer gives you the same address as the address
38
+ * of the second element (a100[1]) of the array.
39
+ *
40
+ * Conclusion: When you add a number to a pointer, the compiler increases
41
+ * the adress by the size of the data type of the pointer. In this case,
42
+ * the data type was an int (4 bytes).
43
+ */
44
+ printf ("ptr1 + 1 = %p\n" , ptr1 + 1 );
45
+ printf ("&a100[1] = %p (expect equal values)\n" , & a10 [1 ]);
46
+ printf ("\n" );
47
+
48
+ /*
49
+ * Let's prove to oursevles that adding 1 to an integer pointer increases
50
+ * the address by 4. To do this, you need to understand that the computer
51
+ * prints out addresses in "hexadecimal" notation. To see how much
52
+ * the address increased, we need to convert the pointer to an unsigned
53
+ * long integer before we do math. To do that, we just add the type
54
+ * to which we want to convert the pointer in parenthesis before the
55
+ * varialbe name. This is called "type casting", because it changes a
56
+ * variable from one type to another.
57
+ *
58
+ * (unsigned long) ptr1;
59
+ *
60
+ * NOTE: A pointer variable is stored in 8 bytes of memory. That's the
61
+ * size of an unsigned long. Eight bytes is 64 bits and
62
+ * 2^64 = 1.8e19 (possible numbers)
63
+ * In laymans terms, that's 18 followed by 18 zeros.
64
+ * 1,000 - thousand (3 zeros)
65
+ * 1,000,000 - million (6 zeros)
66
+ * 1,000,000,000 - billion (9 zeros)
67
+ * 1,000,000,000,000 - trillion (12 zeros)
68
+ * 1,000,000,000,000,000 - quadrillion (15 zeros)
69
+ * 1,000,000,000,000,000,000 - quintillion (18 zeros)
70
+ *
71
+ * So, there are roughly 18 quintillion possible addresses the computer
72
+ * can assign. Since each address is 1 byte, then the toal memory size
73
+ * my computer could address is 1.8e19 bytes. There are 1e12 bytes in a
74
+ * TB (terabyte); so, a modern computer could theoretically address
75
+ *
76
+ * 1.8e19 / 1.0e12 = 1.8e7 TB
77
+ * 18,000,000 (18 Million TB)
78
+ *
79
+ * In 2022, Seagate's popular external backup drives are between 1 and 5 TB;
80
+ * so, you'd need quite a few of those (5 to 18 million of them) to exhaust
81
+ * the address space of your computer.
82
+ *
83
+ * https://www.seagate.com/consumer/backup/backup-plus/#specs
84
+ *
85
+ */
86
+
87
+ printf ("ptr1 + 1 = %p (%020lu)\n" , ptr1 + 1 , (unsigned long ) (ptr1 + 1 ));
88
+ printf ("ptr1 = %p (%020lu)\n" , ptr1 , (unsigned long ) ptr1 );
89
+ printf ("--------------------------------------------- (subtraction)\n" );
90
+ printf (" 0x%09x (%020li) (expect 4)\n" ,
91
+ (unsigned int )((unsigned long ) (ptr1 + 1 ) - (unsigned long ) ptr1 ),
92
+ (unsigned long ) (ptr1 + 1 ) - (unsigned long ) ptr1 );
93
+ printf ("\n" );
94
+
95
+ /*
96
+ * NOTE: The code above is very hard to read. Don't stress too much if you
97
+ * can't understand it easily. In summary, the code converts ptr and (ptr+1)
98
+ * into unsigned long integers so we can perform the math to subtract
99
+ * the adresses represented by ptr and (ptr+1). In the end, we prove that the
100
+ * difference in the addresses is 4.
101
+ */
102
+
103
+ /*
104
+ * Arrays are Pointers!
105
+ * So far we've seen that I can increase the address stored in a pointer
106
+ * by the size of the data type of the pointer simply by adding 1 to the
107
+ * pointer.
108
+ *
109
+ * We've also seen that adding 1 to a pointer gives us the same address
110
+ * as if we indexed the second element of an array.
111
+ *
112
+ * Let's solidify our understanding with some examples of equivalent code.
113
+ */
114
+
115
+ /*
116
+ * Each statement below is quivalent.
117
+ * This means that referencing an index of an array is the same as doing
118
+ * pointer arithmetic on the address of the first element of an array.
119
+ * In the final 2 statement we see that the name of an array IS a pointer
120
+ * and array style indexes can be applied to pointers. Or said another way,
121
+ * a pointer to the address of the first element of an array IS an array!
122
+ */
123
+ a10 [5 ] = 100 ;
124
+ printf ("a100[5] = %i (expect 100)\n" , a10 [5 ]);
125
+ * (ptr1 + 5 ) = 101 ;
126
+ printf ("a100[5] = %i (expect 101)\n" , a10 [5 ]);
127
+ * (a10 + 5 ) = 102 ;
128
+ printf ("a100[5] = %i (expect 102)\n" , a10 [5 ]);
129
+ ptr1 [5 ] = 103 ;
130
+ printf ("a100[5] = %i (expect 103)\n" , a10 [5 ]);
131
+ printf ("\n" );
132
+
133
+ /*
134
+ * Arrays are Contiguous Spaces of Memory.
135
+ *
136
+ * An array is a named area of contiguous memory. The size and type
137
+ * of the array determine the amount of memory the computer assigns.
138
+ * The assigned memory is always one contiguous block.
139
+ *
140
+ * The size of the array defined in the block below is equal to
141
+ *
142
+ * sizeof(int) * 10 = 40 bytes (on most computers)
143
+ *
144
+ * We will prove it to ourselves using pointers. The code below gets a
145
+ * pointer to the first and last element of the array. Then we subtract
146
+ * the memory addresses. The difference should be 9 * sizeof(int) on your
147
+ * computer. Why isn't it 10 * sizeof(int)? It's because subtracting the
148
+ * address of the last element from the first element doesn't account
149
+ * for the final 4 bites referenced by the address of the last element.
150
+ * The lastelement_address is the beginning of a 4 byte block that
151
+ * "starts" at that address.
152
+ *
153
+ * Index Address
154
+ * ------------------ <firstelement_address>
155
+ * 0 | 4 bytes | |
156
+ * ------------------ |
157
+ * 1 | 4 bytes | |
158
+ * ------------------ |
159
+ * 2 | 4 bytes | |
160
+ * ------------------ |
161
+ * 3 | 4 bytes | |
162
+ * ------------------ |
163
+ * 4 | 4 bytes | | --> 36 bytes between the
164
+ * ------------------ | firstelement_address and
165
+ * 5 | 4 bytes | | lastelement_address.
166
+ * ------------------ |
167
+ * 6 | 4 bytes | |
168
+ * ------------------ |
169
+ * 7 | 4 bytes | |
170
+ * ------------------ |
171
+ * 8 | 4 bytes | |
172
+ * ------------------ <lastelement_address>
173
+ * 9 | 4 bytes | | --> These final 4 bytes are not
174
+ * ------------------ counted when I simply subtract
175
+ * the lastelement_address from
176
+ * the firstelement_address.
177
+ */
178
+
179
+ int * firstelement = & a10 [0 ];
180
+ int * lastelement = & a10 [9 ];
181
+ unsigned long memdiff = (unsigned long )lastelement - (unsigned long )firstelement ;
182
+
183
+ printf ("Address of array2[0] = %p\nAddress of array2[9] = %p\n" ,
184
+ firstelement , lastelement );
185
+ printf ("Difference between addresses: %lu\n" , memdiff );
186
+ printf ("\n" );
187
+
188
+ return 1 ;
189
+ }
0 commit comments