114
0 vs NULL // Twitter @raysato Aykut Erdem // Koç University // Fall 2021 COMP201 Computer Systems & Programming Lecture #7 Arrays and Pointers

Computer Systems & Programming

  • Upload
    others

  • View
    9

  • Download
    0

Embed Size (px)

Citation preview

Page 1: Computer Systems & Programming

0 vs NULL // Twitter @raysato

Aykut Erdem // Koç University // Fall 2021

COMP201Computer Systems &Programming

Lecture #7 – Arrays and Pointers

Page 2: Computer Systems & Programming

Good news, everyone!

• Assg1 is due Oct 21

• Assg2 will be out on Oct 21 (due Nov 4)

22

Image: Professor Farnsworth (Futurama)

Page 3: Computer Systems & Programming

Recap: Common string.h FunctionsFunction Description

strlen(str) returns the # of chars in a C string (before null-terminating character).

strcmp(str1, str2), strncmp(str1, str2, n)

compares two strings; returns 0 if identical, <0 if str1 comes before str2 in alphabet, >0 if str1 comes after str2 in alphabet. strncmpstops comparing after at most n characters.

strchr(str, ch)strrchr(str, ch)

character search: returns a pointer to the first occurrence of ch in str, or NULL if ch was not found in str. strrchr find the last occurrence.

strstr(haystack, needle) string search: returns a pointer to the start of the first occurrence of needle in haystack, or NULL if needle was not found in haystack.

strcpy(dst, src),strncpy(dst, src, n)

copies characters in src to dst, including null-terminating character. Assumes enough space in dst. Strings must not overlap. strncpystops after at most n chars, and does not add null-terminating char.

strcat(dst, src),strncat(dst, src, n)

concatenate src onto the end of dst. strncat stops concatenating after at most n characters. Always adds a null-terminating character.

strspn(str, accept),strcspn(str, reject)

strspn returns the length of the initial part of str which contains only characters in accept. strcspn returns the length of the initial part of str which does not contain any characters in reject.

3

Page 4: Computer Systems & Programming

Recap: Pointers• A pointer is a variable that stores a memory address.• Because there is no pass-by-reference in C like in C++, pointers let us

pass around the address of one instance of memory, instead of making many copies.• One (8 byte) pointer can represent any size memory location!• Pointers are also essential for allocating memory on the heap, which we

will cover later.• Pointers also let us refer to memory generically, which we will cover

later.

4

Page 5: Computer Systems & Programming

Recap: Pointersint x = 2;

// Make a pointer that stores the address of x.// (& means "address of")int *xPtr = &x;

// Dereference the pointer to go to that address.// (* means "dereference")printf("%d", *xPtr); // prints 2

5

Page 6: Computer Systems & Programming

Recap: Strings In Memory1. If we create a string as a char[], we can modify its characters because its memory

lives in our stack space.

2. We cannot set a char[] equal to another value, because it is not a pointer; it refers to the block of memory reserved for the original array.

3. If we pass a char[] as a parameter, set something equal to it, or perform arithmetic with it, it’s automatically converted to a char *.

4. If we create a new string with new characters as a char *, we cannot modify its characters because its memory lives in the data segment.

5. We can set a char * equal to another value, because it is a reassign-able pointer.

6. Adding an offset to a C string gives us a substring that many places past the first character.

7. If we change characters in a string parameter, these changes will persist outside of the function.

6

Page 7: Computer Systems & Programming

Recap: Character ArraysAddress Value

0x105 '\0'

0x104 'e'

0x103 'l'

0x102 'p'

0x101 'p'

0x100 'a'

When we declare an array of characters, contiguous memory is allocated on the stack to store the contents of the entire array. We can modify what is on the stack.

char str[6];strcpy(str, "apple");

str

STACK

7

Page 8: Computer Systems & Programming

Recap: Recap: char *

Address Value…

0xff0 0x10……

0x12 '\0'0x11 'i'0x10 'h'

When we declare a char pointer equal to a string literal, the characters are not stored on the stack. Instead, they are stored in a special area of memory called the “data segment”. We cannot modify memory in this segment.char *str = "hi";The pointer variable (e.g. str) refers to theaddress of the first character of the string in the data segment.

strSTACK

DATA SEGMENT

This applies only to creating newstrings with char *. This does notapply for making a char * that points to an existing stack string.

8

Page 9: Computer Systems & Programming

Plan for Today• Strings in Memory (cont’d.)• Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

Disclaimer: Slides for this lecture were borrowed from —Nick Troccoli and Lisa Yan's Stanford CS107 class

9

Page 10: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

10

Page 11: Computer Systems & Programming

Memory LocationsFor each code snippet below, can we modify the characters in myStr?

char myStr[6];

Key Question: where do its characters live? Do they live in memory we own? Or the read-only data segment?

11

Page 12: Computer Systems & Programming

Memory LocationsFor each code snippet below, can we modify the characters in myStr?

char *myStr = "Hi";

12

Key Question: where do its characters live? Do they live in memory we own? Or the read-only data segment?

Page 13: Computer Systems & Programming

Memory LocationsFor each code snippet below, can we modify the characters in myStr?

char buf[6];strcpy(buf, "Hi");char *myStr = buf;

Key Question: where do its characters live? Do they live in memory we own? Or the read-only data segment?

13

Page 14: Computer Systems & Programming

Memory LocationsFor each code snippet below, can we modify the characters in myStr?

char *otherStr = "Hi";char *myStr = otherStr;

Key Question: where do its characters live? Do they live in memory we own? Or the read-only data segment?

14

Page 15: Computer Systems & Programming

Memory LocationsFor each code snippet below, can we modify the characters in myStr?

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char buf[6];strcpy(buf, "Hi");myFunc(buf);return 0;

}

Key Question: where do its characters live? Do they live in memory we own? Or the read-only data segment?

15

Page 16: Computer Systems & Programming

Memory LocationsQ: Is there a way to check in code whether a string’s characters are modifiable?A: No. This is something you can only tell by looking at the code itself and how the string was created.

Q: So then if I am writing a string function that modifies a string, how can I tell if the string passed in is modifiable?A: You can’t! This is something you instead state as an assumption in your function documentation. If someone calls your function with a read-only string, it will crash, but that’s not your function’s fault :-)

16

Page 17: Computer Systems & Programming

String Behavior #5: We can set a char *equal to another value, because it is a reassign-able pointer.

17

Page 18: Computer Systems & Programming

char *A char * variable refers to a single character. We can reassign an existing char * pointer to be equal to another char * pointer.

char *str = "apple"; // e.g. 0xfff0char *str2 = "apple 2"; // e.g. 0xfe0str = str2; // ok! Both store address 0xfe0

18

Page 19: Computer Systems & Programming

Arrays and PointersWe can also make a pointer equal to an array; it will point to the first element in that array.

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");char *ptr = str;...

}

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'0xf8 0x100

strptr

main()

STACK

19

Page 20: Computer Systems & Programming

Arrays and PointersWe can also make a pointer equal to an array; it will point to the first element in that array.

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");char *ptr = str;

// equivalentchar *ptr = &str[0];

// confusingly equivalent, avoidchar *ptr = &str;...

}

STACKAddress Value

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'0xf8 0x100

strptr

main()

20

Page 21: Computer Systems & Programming

String Behavior #6: Adding an offset to a C string gives us a substring that many places past the first character.

21

Page 22: Computer Systems & Programming

Pointer ArithmeticWhen we do pointer arithmetic, we are adjusting the pointer by a certain number of places (e.g. characters).

char *str = "apple"; // e.g. 0xff0char *str2 = str + 1; // e.g. 0xff1char *str3 = str + 3; // e.g. 0xff3

printf("%s", str); // appleprintf("%s", str2); // ppleprintf("%s", str3); // le

Address Value

0xff5 '\0'

0xff4 'e'

0xff3 'l'

0xff2 'p'

0xff1 'p'

0xff0 'a'

TEXT SEGMENT

22

Page 23: Computer Systems & Programming

char *When we use bracket notation with a pointer, we are performing pointer arithmetic and dereferencing:

char *str = "apple"; // e.g. 0xff0

// both of these add three places to str,// and then dereference to get the char there.// E.g. get memory at 0xff3.char thirdLetter = str[3]; // 'l'char thirdLetter = *(str + 3); // 'l'

Address Value

0xff5 '\0'

0xff4 'e'

0xff3 'l'

0xff2 'p'

0xff1 'p'

0xff0 'a'

TEXT SEGMENT

23

Page 24: Computer Systems & Programming

String Behavior #7: If we change characters in a string parameter, these changes will persist outside of the function.

24

Page 25: Computer Systems & Programming

Strings as ParametersWhen we pass a char * string as a parameter, C makes a copy of the address stored in the char * and passes it to the function. This means they both refer to the same memory location.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char *str = "apple";myFunc(str);...

}

Address Value

0xfff0 0x10

0xff0 0x10

str

myStr

main()

myFunc()

STACK

25

Page 26: Computer Systems & Programming

Strings as ParametersWhen we pass a char array as a parameter, C makes a copy of the address of the first array elementand passes it (as a char *) to the function.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);...

}

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

26

Page 27: Computer Systems & Programming

Strings as ParametersWhen we pass a char array as a parameter, C makes a copy of the address of the first array elementand passes it (as a char *) to the function.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");// equivalentchar *strAlt = str;myFunc(strAlt);...

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

27

Page 28: Computer Systems & Programming

Strings as ParametersThis means if we modify characters in myFunc, the changes will persist back in main!

void myFunc(char *myStr) {myStr[4] = 'y';

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);printf("%s", str); // apply...

}

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

28

Page 29: Computer Systems & Programming

Strings as ParametersThis means if we modify characters in myFunc, the changes will persist back in main!

void myFunc(char *myStr) {myStr[4] = 'y';

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);printf("%s", str); // apply...

}

Address Value…

0x105 '\0'0x104 'y'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

29

Page 30: Computer Systems & Programming

Strings In Memory1. If we create a string as a char[], we can modify its characters because its memory

lives in our stack space.

2. We cannot set a char[] equal to another value, because it is not a pointer; it refers to the block of memory reserved for the original array.

3. If we pass a char[] as a parameter, set something equal to it, or perform arithmetic with it, it’s automatically converted to a char *.

4. If we create a new string with new characters as a char *, we cannot modify its characters because its memory lives in the data segment.

5. We can set a char * equal to another value, because it is a reassign-able pointer.

6. Adding an offset to a C string gives us a substring that many places past the first character.

7. If we change characters in a string parameter, these changes will persist outside of the function.

30

Page 31: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

31

Page 32: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");Compile error (cannot reassign array)

32

Page 33: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

33

Compile error (cannot reassign array)

Page 34: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

Compile error (cannot reassign array)Segmentation fault (string literal)

34

Page 35: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

3. char arr[7];strcpy(arr, "Hello3");char *str = arr;

35

Compile error (cannot reassign array)Segmentation fault (string literal)

Page 36: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

3. char arr[7];strcpy(arr, "Hello3");char *str = arr;

Compile error (cannot reassign array)Segmentation fault (string literal)

Prints eulo336

Page 37: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

3. char arr[7];strcpy(arr, "Hello3");char *str = arr;

4. char *ptr = "Hello4";char *str = ptr;

37

Compile error (cannot reassign array)Segmentation fault (string literal)

Prints eulo3

Page 38: Computer Systems & Programming

char* vs char[] exercisesSuppose we use avariable str as follows:

For each of thefollowing instantiations:

• Will there be a compile error/segfault?• If no errors, what is printed?

str = str + 1;str[1] = 'u';printf("%s", str);

🤔

1. char str[7];strcpy(str, "Hello1");

2. char *str = "Hello2";

3. char arr[7];strcpy(arr, "Hello3");char *str = arr;

4. char *ptr = "Hello4";char *str = ptr;

Compile error (cannot reassign array)Segmentation fault (string literal)

Prints eulo3 Segmentation fault (string literal)38

Page 39: Computer Systems & Programming

COMP201 Topic 4: How can we effectively manage all types of

memory in our programs?

Page 40: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

40

Page 41: Computer Systems & Programming

C ParametersWhen you pass a value as a parameter, C passes a copy of that value.

void myFunction(int x) {…

}

int main(int argc, char *argv[]) {int num = 4;myFunction(num); // passes copy of 4

}

41

Page 42: Computer Systems & Programming

C ParametersWhen you pass a value as a parameter, C passes a copy of that value.

void myFunction(int *x) {…

}

int main(int argc, char *argv[]) {int num = 4;myFunction(&num); // passes copy of e.g. 0xffed63

}

42

Page 43: Computer Systems & Programming

C ParametersWhen you pass a value as a parameter, C passes a copy of that value.

void myFunction(char ch) {…

}

int main(int argc, char *argv[]) {char *myStr = "Hello!"; myFunction(myStr[1]); // passes copy of 'e'

}

43

Page 44: Computer Systems & Programming

C ParametersIf you are performing an operation with some input and do not care about any changes to the input, pass the data type itself.

44

Page 45: Computer Systems & Programming

C ParametersIf you are performing an operation with some input and do not care about any changes to the input, pass the data type itself.

void myFunction(char ch) {printf("%c", ch);

}

int main(int argc, char *argv[]) {char *myStr = "Hello!"; myFunction(myStr[1]); // prints 'e'

}

45

Page 46: Computer Systems & Programming

C ParametersIf you are performing an operation with some input and do not care about any changes to the input, pass the data type itself.

int myFunction(int num1, int num2) {return x + y;

}

int main(int argc, char *argv[]) {int x = 5;int y = 6;int sum = myFunction(x, y); // returns 11

}46

Page 47: Computer Systems & Programming

C ParametersIf you are modifying a specific instance of some value, pass the location of what you would like to modify.

Do I care about modifying this instance of my data? If so, I need to pass where that instance lives, as a parameter, so it can be modified.

47

Page 48: Computer Systems & Programming

PointersIf you are modifying a specific instance of some value, pass the location of what you would like to modify.

void capitalize(char *ch) {// modifies what is at the address stored in ch

}

int main(int argc, char *argv[]) {char letter = 'h';/* We don’t want to capitalize any instance of 'h'.* We want to capitalize *this* instance of 'h'! */

capitalize(&letter);printf("%c", letter); // want to print 'H';

} 48

Page 49: Computer Systems & Programming

PointersIf you are modifying a specific instance of some value, pass the location of what you would like to modify.

void doubleNum(int *x) {// modifies what is at the address stored in x

}

int main(int argc, char *argv[]) {int num = 2;/* We don’t want to double any instance of 2.* We want to double *this* instance of 2! */doubleNum(&num);printf("%d", num); // want to print 4;

}49

Page 50: Computer Systems & Programming

PointersIf a function takes an address (pointer) as a parameter, it can go to that address if it needs the actual value.

void capitalize(char *ch) {// *ch gets the character stored at address ch.char newChar = toupper(*ch);

// *ch = goes to address ch and puts newChar there.*ch = newChar;

}

50

Page 51: Computer Systems & Programming

PointersIf a function takes an address (pointer) as a parameter, it can go to that address if it needs the actual value.

void capitalize(char *ch) {/* go to address ch and put the capitalized version* of what is at address ch there. */*ch = toupper(*ch);

}

51

Page 52: Computer Systems & Programming

PointersIf a function takes an address (pointer) as a parameter, it can go to that address if it needs the actual value.

void capitalize(char *ch) {// this capitalizes the address ch! Lchar newChar = toupper(ch);

// this stores newChar in ch as an address! Lch = newChar;

}

52

Page 53: Computer Systems & Programming

char *• A char * is technically a pointer to a single character.• We commonly use char * as string by having the character it points

to be followed by more characters and ultimately a null terminator.• A char * could also just point to a single character (not a string).

53

Page 54: Computer Systems & Programming

String Behavior #7: If we change characters in a string parameter, these changes will persist outside of the function.

54

Page 55: Computer Systems & Programming

Strings as ParametersWhen we pass a char * string as a parameter, C makes a copy of the address stored in the char *, and passes it to the function. This means they both refer to the same memory location.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char *str = "apple";myFunc(str);...

}

Address Value

0xfff0 0x10

0xff0 0x10

str

myStr

main()

myFunc()

STACK

55

Page 56: Computer Systems & Programming

Strings as ParametersWhen we pass a char array as a parameter, C makes a copy of the address of the first array element, and passes it (as a char *) to the function.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);...

}

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

56

Page 57: Computer Systems & Programming

Strings as ParametersWhen we pass a char array as a parameter, C makes a copy of the address of the first array element, and passes it (as a char *) to the function.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");// equivalentchar *strAlt = str;myFunc(strAlt);...

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

57

Page 58: Computer Systems & Programming

Strings as ParametersThis means if we modify characters in myFunc, the changes will persist back in main!

void myFunc(char *myStr) {myStr[4] = 'y';

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);printf("%s", str); // apply...

}

Address Value…

0x105 '\0'0x104 'e'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

58

Page 59: Computer Systems & Programming

Strings as ParametersThis means if we modify characters in myFunc, the changes will persist back in main!

void myFunc(char *myStr) {myStr[4] = 'y';

}

int main(int argc, char *argv[]) {char str[6];strcpy(str, "apple");myFunc(str);printf("%s", str); // apply...

}

Address Value…

0x105 '\0'0x104 'y'0x103 'l'0x102 'p'0x101 'p'0x100 'a'

……

0xf 0x100…

str

myStr

main()

myFunc()

STACK

59

Page 60: Computer Systems & Programming

Exercise 1We want to write a function that prints out the square of a number. What should go in each of the blanks?

void printSquare(__?__) {int square = __?__ * __?__;printf("%d", square);

}

int main(int argc, char *argv[]) {int num = 3;printSquare(__?__); // should print 9

}60

Page 61: Computer Systems & Programming

Exercise 1We want to write a function that prints out the square of a number. What should go in each of the blanks?

void printSquare(int x) {int square = x * x;printf("%d", square);

}

int main(int argc, char *argv[]) {int num = 3;printSquare(num); // should print 9

}

We are performing a calculation with some input and do not care about any changes to the input, so we pass the data type itself.

61

Page 62: Computer Systems & Programming

Exercise 1We want to write a function that prints out the square of a number. What should go in each of the blanks?

void printSquare(int x) {x = x * x;printf("%d", x);

}

int main(int argc, char *argv[]) {int num = 3;printSquare(num); // should print 9

}

We are performing a calculation with some input and do not care about any changes to the input, so we pass the data type itself.

62

Page 63: Computer Systems & Programming

Exercise 2We want to write a function that flips the case of a letter. What should go in each of the blanks?

void flipCase(__?__) {if (isupper(__?__)) {

__?__ = __?__;} else if (islower(__?__)) {

__?__ = __?__;}

}

int main(int argc, char *argv[]) {char ch = 'g';flipCase(__?__);printf("%c", ch); // want this to print ‘G’

} 63

Page 64: Computer Systems & Programming

Exercise 2We want to write a function that flips the case of a letter. What should go in each of the blanks?

void flipCase(char *letter) {if (isupper(*letter)) {

*letter = tolower(*letter);} else if (islower(*letter)) {

*letter = toupper(*letter);}

}

int main(int argc, char *argv[]) {char ch = 'g';flipCase(&ch);printf("%c", ch); // want this to print ‘G’

}

We are modifying a specific instance of the letter, so we pass the location of the letter we would like to modify.

64

Page 65: Computer Systems & Programming

Pointers Summary• If you are performing an operation with some input and do not care

about any changes to the input, pass the data type itself.• If you are modifying a specific instance of some value, pass the

location of what you would like to modify.• If a function takes an address (pointer) as a parameter, it can go to that

address if it needs the actual value.

65

Page 66: Computer Systems & Programming

Pointers Summary• Tip: setting a function parameter equal to a new value usually doesn’t do

what you want. Remember that this is setting the function’s own copy of the parameter equal to some new value.

void doubleNum(int x) {x = x * x; // modifies doubleNum’s own copy!

}

void advanceStr(char *str) {str += 2; // modifies advanceStr’s own copy!

}

66

Page 67: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

67

Page 68: Computer Systems & Programming

Exercise 3Sometimes, we would like to modify a string’s pointer itself, rather than just the characters it points to. E.g. we want to write a function skipSpaces that modifies a string pointer to skip past any initial spaces. What should go in each of the blanks?

void skipSpaces(__?__) {...

}

int main(int argc, char *argv[]) {char *str = " hello";skipSpaces(__?__);printf("%s", str); // should print "hello"

}68

Page 69: Computer Systems & Programming

Exercise 3Sometimes, we would like to modify a string’s pointer itself, rather than just the characters it points to. E.g. we want to write a function skipSpaces that modifies a string pointer to skip past any initial spaces. What should go in each of the blanks?

void skipSpaces(char **strPtr) {...

}

int main(int argc, char *argv[]) {char *str = " hello";skipSpaces(&str);printf("%s", str); // should print "hello"

}

We are modifying a specific instance of the string pointer, so we pass the location of the string pointer we would like to modify.

69

Page 70: Computer Systems & Programming

Exercise 3Sometimes, we would like to modify a string’s pointer itself, rather than just the characters it points to. E.g. we want to write a function skipSpaces that modifies a string pointer to skip past any initial spaces. What should go in each of the blanks?

void skipSpaces(char *strPtr) {...

}

int main(int argc, char *argv[]) {char *str = " hello";skipSpaces(str);printf("%s", str); // should print "hello"

}

This advances skipSpace’s own copy of the string pointer, not the instance in main.

70

Page 71: Computer Systems & Programming

Demo: Skip Spaces

skip_spaces.c

Page 72: Computer Systems & Programming

Pointers to Stringsvoid skipSpaces(char **strPtr) {

int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

Address Value……

main()STACK

72

Page 73: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0xf

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

myStr

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

main()STACK

73

Page 74: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0xf

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

myStr

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

main()STACK

74

Page 75: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0xf

……

0xf0 0x105……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStr

strPtr

main()

skipSpaces()

STACK

75

Page 76: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0xf

……

0xf0 0x1050xe8 2

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStr

strPtrnumSpaces

main()

skipSpaces()

STACK

76

Page 77: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0xf

……

0xf0 0x1050xe8 2

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStr

strPtrnumSpaces

main()

skipSpaces()

STACK

77

Page 78: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0x11

……

0xf0 0x1050xe8 2

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStr

strPtrnumSpaces

main()

STACK

skipSpaces()

78

Page 79: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0x11

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStrSTACK main()

79

Page 80: Computer Systems & Programming

Pointers to StringsAddress Value

…0x105 0x11

……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char **strPtr) {int numSpaces = strspn(*strPtr, " ");*strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(&myStr);printf("%s\n", myStr); // hireturn 0;

}

myStrmain()STACK

80

Page 81: Computer Systems & Programming

Making CopiesAddress Value

…0x105 0xf

……

0xf0 0xf……

0x13 '\0'0x12 'i'0x11 'h'0x10 ' '0xf ' '

DATA SEGMENT

void skipSpaces(char *strPtr) {int numSpaces = strspn(strPtr, " ");strPtr += numSpaces;

}

int main(int argc, char *argv[]) {char *myStr = " hi";skipSpaces(myStr);printf("%s\n", myStr); // hireturn 0;

}

myStr

strPtr

main()

skipSpaces()

STACK

81

myFunc myFunc

Page 82: Computer Systems & Programming

Skip spacesvoid skip_spaces(char **p_str) {

int num = strspn(*p_str, " ");*p_str = *p_str + num;

}int main(int argc, char *argv[]){

char *str = " Hi!";skip_spaces(&str);printf("%s", str); // "Hi!"return 0;

}

🤔

123456789

10

What diagram most accurately depicts program state at Line 4 (before skip_spaces returns to main)?

str

' ' ' ' 'H' 'i' '!' '\0'

p_str

strp_str

' ' ' ' 'H' 'i' '!' '\0'

str

' ' ' ' 'H' 'i' '!' '\0'

p_str

A.

B.

C.

82

Page 83: Computer Systems & Programming

Skip spaces

str

' ' ' ' 'H' 'i' '!' '\0'

p_str

strp_str

' ' ' ' 'H' 'i' '!' '\0'

str

' ' ' ' 'H' 'i' '!' '\0'

p_str

A.

B.

C.

void skip_spaces(char **p_str) {int num = strspn(*p_str, " ");*p_str = *p_str + num;

}int main(int argc, char *argv[]){

char *str = " Hi!";skip_spaces(&str);printf("%s", str); // "Hi!"return 0;

}

123456789

10

What diagram most accurately depicts program state at Line 4 (before skip_spaces returns to main)?

83

Page 84: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

84

Page 85: Computer Systems & Programming

ArraysAddress Value

0x105 '\0'

0x104 'e'

0x103 'l'

0x102 'p'

0x101 'p'

0x100 'a'

When you declare an array, contiguous memory is allocated on the stack to store the contents of the entire array.

char str[6];strcpy(str, "apple");

The array variable (e.g. str) is not a pointer; it refers to the entire array contents. In fact, sizeof returns the size of the entire array!

int arrayBytes = sizeof(str); // 6

str

STACK

85

Page 86: Computer Systems & Programming

ArraysAn array variable refers to an entire block of memory. You cannot reassign an existing array to be equal to a new array.

int nums[] = {1, 2, 3};int nums2[] = {4, 5, 6, 7};nums = nums2; // not allowed!

An array’s size cannot be changed once you create it; you must create another new array instead.

86

Page 87: Computer Systems & Programming

Arrays as ParametersWhen you pass an array as a parameter, C makes a copy of the address of the first array element, and passes it (a pointer) to the function.

void myFunc(char *myStr) {...

}

int main(int argc, char *argv[]) {char str[3];strcpy(str, "hi");myFunc(str);...

}

Address Value0x1f2 '\0'

0x1f1 'i'

0x1f0 'h'...

0xff

0xfe

0xfd

0xfc

0xfb

0xfa

0xf9

0xf8...

str

mystr

main()

STACK

0x1f0myFunc()

87

Page 88: Computer Systems & Programming

Arrays as ParametersWhen you pass an array as a parameter, C makes a copy of the address of the first array element and passes it (a pointer) to the function.

void myFunc(char *myStr) {…

}

int main(int argc, char *argv[]) {char str[3];strcpy(str, "hi");// equivalentchar *arrPtr = str;myFunc(arrPtr);...

}

Address Value0x1f2 '\0'0x1f1 'i'0x1f0 'h'0x1e8 0x1f0

0x10 0x1f0…

str

myStr

main()

STACK

arrPtr

myFunc()

88

Page 89: Computer Systems & Programming

Arrays as ParametersThis also means we can no longer get the full size of the array using sizeof, because now it is just a pointer.

void myFunc(char *myStr) {int size = sizeof(myStr); // 8

}

int main(int argc, char *argv[]) {char str[3];strcpy(str, "hi");int size = sizeof(str); // 3myFunc(str);...

}

Address Value0x1f2 '\0'

0x1f1 'i'

0x1f0 'h'...

0xff

0xfe

0xfd

0xfc

0xfb

0xfa

0xf9

0xf8

str

mystr

main()

STACK

0x1f0myFunc()

89

Page 90: Computer Systems & Programming

sizeof returns the size of an array, or 8 for a pointer. Therefore, when we pass an array as a parameter, we can no longer use sizeof to get its full size.

90

Page 91: Computer Systems & Programming

Arrays and PointersYou can also make a pointer equal to an array; it will point to the first element in that array.

int main(int argc, char *argv[]) {char str[3];strcpy(str, "hi");char *ptr = str;...

}

Address Value0x1f2 '\0'0x1f1 'i'0x1f0 'h'0x1ef0x1ee0x1ed0x1ec0x1eb0x1ea0x1e90x1e8

str

ptr

main()

STACK

0x1f0

91

Page 92: Computer Systems & Programming

Arrays and PointersYou can also make a pointer equal to an array; it will point to the first element in that array.

int main(int argc, char *argv[]) {char str[3];strcpy(str, "hi");char *ptr = str;

// equivalentchar *ptr = &str[0];

// equivalent, but avoidchar *ptr = &str;...

}

Address Value0x1f2 '\0'0x1f1 'i'0x1f0 'h'0x1ef0x1ee0x1ed0x1ec0x1eb0x1ea0x1e90x1e8

str

ptr

main()

STACK

0x1f0

92

Page 93: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

93

Page 94: Computer Systems & Programming

Arrays Of PointersYou can make an array of pointers to e.g. group multiple strings together:

char *stringArray[5]; // space to store 5 char *s

This stores 5 char *s, not all of the characters for 5 strings!

char *str0 = stringArray[0]; // first char *

94

Page 95: Computer Systems & Programming

Arrays Of Pointers./swapwords apple banana orange peach pear

95

Page 96: Computer Systems & Programming

Arrays Of Pointers./swapwords apple banana orange peach pear

What is the value of argv[2] in this diagram?

96

Page 97: Computer Systems & Programming

Lecture Plan• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

97

Page 98: Computer Systems & Programming

Pointer ArithmeticWhen you do pointer arithmetic, you are adjusting the pointer by a certain number of places (e.g. characters).

char *str = "apple"; // e.g. 0xff0char *str1 = str + 1; // e.g. 0xff1char *str3 = str + 3; // e.g. 0xff3

printf("%s", str); // appleprintf("%s", str1); // ppleprintf("%s", str3); // le

Address Value

0xff5 '\0'

0xff4 'e'

0xff3 'l'

0xff2 'p'

0xff1 'p'

0xff0 'a'

DATA SEGMENT

98

Page 99: Computer Systems & Programming

Pointer ArithmeticPointer arithmetic does not work in bytes. Instead, it works in the size of the type it points to.

// nums points to an int arrayint *nums = … // e.g. 0xff0int *nums1 = nums + 1; // e.g. 0xff4int *nums3 = nums + 3; // e.g. 0xffc

printf("%d", *nums); // 52printf("%d", *nums1); // 23printf("%d", *nums3); // 34

Address Value

0x1004 1

0x1000 16

0xffc 34

0xff8 12

0xff4 23

0xff0 52

STACK

99

Page 100: Computer Systems & Programming

Pointer ArithmeticWhen you use bracket notation with a pointer, you are actually performing pointer arithmetic and dereferencing:

char *str = "apple";// e.g. 0xff0

// both of these add two places to str,// and then dereference to get the char there.// E.g. get memory at 0xff2.char thirdLetter = str[2]; // 'p'char thirdLetter = *(str + 2); // 'p'

Address Value

0xff5 '\0'

0xff4 'e'

0xff3 'l'

0xff2 'p'

0xff1 'p'

0xff0 'a'

DATA SEGMENT

100

Page 101: Computer Systems & Programming

Pointer ArithmeticPointer arithmetic with two pointers does not give the byte difference. Instead, it gives the number of places they differ by.

// nums points to an int arrayint *nums = … // e.g. 0xff0int *nums3 = nums + 3; // e.g. 0xffcint diff = nums3 - nums; // 3

Address Value

0x1004 1

0x1000 16

0xffc 34

0xff8 12

0xff4 23

0xff0 52

STACK

101

Page 102: Computer Systems & Programming

String Behavior #6: Adding an offset to a C string gives us a substring that many places past the first character.

102

Page 103: Computer Systems & Programming

How does the code know how many bytes it should look at once it visits an address?

int x = 2;int *xPtr = &x; // e.g. 0xff0

// How does it know to print out just the 4 bytes at xPtr?printf("%d", *xPtr); // 2

Pointer Arithmetic

103

Page 104: Computer Systems & Programming

How does the code know how many bytes it should add when performing pointer arithmetic?

int nums[] = {1, 2, 3};

// How does it know to add 4 bytes here?int *intPtr = nums + 1;

char str[6];strcpy(str, "COMP201");

// How does it know to add 1 byte here?char *charPtr = str + 1;

Pointer Arithmetic

104

Page 105: Computer Systems & Programming

• At compile time, C can figure out the sizes of different data types, and the sizes of what they point to.• For this reason, when the program runs, it knows the correct number of

bytes to address or add/subtract for each data type.

Pointer Arithmetic

105

Page 106: Computer Systems & Programming

Pointer arithmeticArray indexing is “syntactic sugar” for pointer arithmetic:

ptr + i ó &ptr[i]*(ptr + i) ó ptr[i]

• This means too-large/negative subscripts will compile Jarr[99] arr[-1]

• You can use either syntax on either pointer or array.

⚠ Pointer arithmetic does not work in bytes; it works on the type it points to.On int* addresses scale by sizeof(int), on char* scale by sizeof(char).

106

Page 107: Computer Systems & Programming

Extra Slides

Page 108: Computer Systems & Programming

void func(char *str) {str[0] = 'S';str++; *str = 'u';str = str + 3;str[-2] = 'm';

}

int main(int argc, const char *argv[]) { char buf[] = "Monday"; printf("before func: %s\n", buf); func(buf); printf("after func: %s\n", buf); return 0;

}

1. Pointer arithmetic1234567

89

1011121314

🤔• Draw memory diagrams!• Pointers store addresses! Make up addresses if it helps your mental model.

• Will there be a compile error/segfault?• If no errors, what is printed?

Page 109: Computer Systems & Programming

void func(char *str) {str[0] = 'S';str++; *str = 'u';str = str + 3;str[-2] = 'm';

}

int main(int argc, const char *argv[]) { char buf[] = "Monday"; printf("before func: %s\n", buf); func(buf);printf("after func: %s\n", buf); return 0;

}

1. Pointer arithmetic

• Draw memory diagrams!• Pointers store addresses! Make up addresses if it helps your mental model.

1234567

89

1011121314

buf

func

str

main

0x60 0x61 0x62 0x63 0x64 0x65 0x66

Page 110: Computer Systems & Programming

2. Code study: strncpy

char *strncpy(char *dest, const char *src, size_t n) {size_t i;for (i = 0; i < n && src[i] != '\0'; i++)

dest[i] = src[i];for ( ; i < n; i++)

dest[i] = '\0';return dest;

}

12345678

0x60 0x61 0x62 0x63 0x64 0x65 0x66

'M' 'o' 'n' 'd' 'a' 'y' '\0'

0x58 0x59 0x5a 0x5b

'F' 'r' 'i' '\0'

buf

str

What happens if we call strncpy(buf, str, 5);?🤔

Page 111: Computer Systems & Programming

2. Code study: strncpy

char *strncpy(char *dest, const char *src, size_t n) {size_t i;for (i = 0; i < n && src[i] != '\0'; i++)

dest[i] = src[i];for ( ; i < n; i++)

dest[i] = '\0';return dest;

}

12345678

dest

0x60 0x61 0x62 0x63 0x64 0x65 0x66

'M' 'o' 'n' 'd' 'a' 'y' '\0'

0x58 0x59 0x5a 0x5b

'F' 'r' 'i' '\0'

src

i

buf

str

What happens if we call strncpy(buf, str, 5);?

5n

Page 112: Computer Systems & Programming

3. Bonus: Tricky addressesvoid tricky_addresses() {char buf[] = "Local";char *ptr1 = buf;char **double_ptr = &ptr1;printf("ptr1's value: %p\n", ptr1);printf("ptr1’s deref : %c\n", *ptr1);printf(" address: %p\n", &ptr1);printf("double_ptr value: %p\n", double_ptr);printf("buf's address: %p\n", &buf);

char *ptr2 = &buf;printf("ptr2's value: %s\n", ptr2);

}

123456789

101112

What is stored in each variable? (We cover double pointers later in the course)

🤔

Page 113: Computer Systems & Programming

3. Bonus: Tricky addressesvoid tricky_addresses() {char buf[] = "Local";char *ptr1 = buf;char **double_ptr = &ptr1;printf("ptr1's value: %p\n", ptr1);printf("ptr1’s deref : %c\n", *ptr1);printf(" address: %p\n", &ptr1);printf("double_ptr value: %p\n", double_ptr);printf("buf's address: %p\n", &buf);

char *ptr2 = &buf;printf("ptr2's value: %s\n", ptr2);

}

123456789

101112

ptr1

0x10

0x18double_ptr

ptr2

0x20

0x28 0x29 0x2a 0x2b 0x2c 0x2d

'L' 'o' 'c' 'a' 'l' '\0'buf

While Line 10 raises a compiler warning, functionally it will still work—because pointers are addresses.

Page 114: Computer Systems & Programming

Recap• Strings in Memory (cont’d.) • Pointers and Parameters• Double Pointers• Arrays in Memory• Arrays of Pointers• Pointer Arithmetic

Next Time: dynamically allocated memory114