Skip to content

Commit bfaa68d

Browse files
committed
Write program 11 for pointer arithmetic and a challenge program
1 parent e3fc5ba commit bfaa68d

File tree

3 files changed

+265
-58
lines changed

3 files changed

+265
-58
lines changed

11-arrays-pointers.c

-58
This file was deleted.

11-challenge.c

+76
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,76 @@
1+
#include <stdio.h>
2+
#include <stdlib.h>
3+
#include <time.h>
4+
5+
#define SIZE 20
6+
7+
/*
8+
* Program 11 : Challenge - Write a function that displays the memory of an array
9+
* The function will receive an integer array and its size then print out
10+
* a table to the screen which looks like this:
11+
*
12+
* Idx Mem Adr b1 b2 b3 b4
13+
* --------------
14+
* [000] | 0x02e192c9 | 0x16b4f7468 c9 92 e1 02
15+
* --------------
16+
* [001] | 0x276bcb99 | 0x16b4f746c 99 cb 6b 27
17+
* --------------
18+
* [002] | 0x1603be07 | 0x16b4f7470 07 be 03 16
19+
* --------------
20+
* [003] | 0x4fb0c8db | 0x16b4f7474 db c8 b0 4f
21+
* --------------
22+
*
23+
* Column 1 (Idx) - the index of the element
24+
* Column 2 (Mem) - the content of the 4 byte block starting at the address
25+
* of the index in hexidecimal (printf format %x)
26+
* Column 3 (Adr) - the address of the first byte of the 4 byte block
27+
* Colunn 4 (b1-b4) - the contents of each byte of the 4 byte integer in hex
28+
* where byte 1 is the byte addressed by the address of the
29+
* index.
30+
*/
31+
32+
/*
33+
* Function printmem
34+
* This function takes an integer array and the size of the array.
35+
* It prints a memory table to the screen.
36+
*/
37+
void printmem (int a[], int size){
38+
39+
printf(" Idx Mem Adr b1 b2 b3 b4\n");
40+
printf(" --------------\n");
41+
for(int i = 0; i < size; i++){
42+
/*
43+
* For each element of the int array, initialize a pointer with the
44+
* address of the index. Cast to a 1 byte pointer (an unsigned char).
45+
* Use pointer arithmetic to get the address of each component byte of
46+
* the 4-byte integer. Then dereference to get the value stored in
47+
* that byte.
48+
*/
49+
unsigned char* ptr = (unsigned char*) &a[i];
50+
unsigned char byte1 = *ptr;
51+
unsigned char byte2 = *(ptr+1);
52+
unsigned char byte3 = *(ptr+2);
53+
unsigned char byte4 = *(ptr+3);
54+
55+
printf("[%03i] | 0x%08x | %p %02x %02x %02x %02x\n",
56+
i, a[i], &a[i], byte1, byte2, byte3, byte4);
57+
printf(" --------------\n");
58+
}
59+
printf("\n");
60+
}
61+
62+
int main(){
63+
/* Construct and initialize an array of size "SIZE" */
64+
int a[SIZE] = {0};
65+
66+
srand(time(NULL));
67+
68+
/* Loop and initialize the array with random numbers. */
69+
for(int i = 0; i < SIZE; i++){
70+
a[i] = rand();
71+
}
72+
73+
/* Call the printmem function */
74+
printmem(a, SIZE);
75+
return 1;
76+
}

11-pointer-arithmetic.c

+189
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,189 @@
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

Comments
 (0)