Skip to content

Latest commit

 

History

History
 
 

lab-04

Folders and files

NameName
Last commit message
Last commit date

parent directory

..
 
 

Lab 04: References and functions

Welcome to CSC 211 Lab 04. Your goal for this lab will be to gain a better understanding of references and functional programming. Be sure to read and follow all instructions unless otherwise specified. Some of the language used here has been adopted from the OpenDSA Data Structures and Algorithms Modules Collection text text. Create a lab-04.txt document to record all of your lab answers in and implement all of your .cpp programs in your IDE.

  1. References [50 minutes]
    1.1 Understanding References
    1.2 Using References in Functions
    1.3 Code Tracing and Conceptual Questions
    1.4 Problem Set
  2. The Call Stack
    2.1 Stack Frames
    2.2 Getting Started With the Stack
  3. Functions [50 minutes]
    3.1 Examples
    3.2 Order Matters
    3.3 Problem Set
  4. Submission [5 minutes]

Part 1.1 Understanding References

References in C++ are an alternative name for an existing variable. They provide a way to create an alias, so you can refer to the same variable by different names. A reference must be initialized when it is created and cannot be changed to reference another variable.

Example

#include <iostream>

int main() {
    int a = 5;
    int& ref = a;  // ref is a reference to a
    std::cout << "a: " << a << ", ref: " << ref << std::endl;

    ref = 10;  // changing ref changes a
    std::cout << "a: " << a << ", ref: " << ref << std::endl;

    return 0;
}

In this example, ref is a reference to a. Modifying ref will modify a because they both refer to the same memory location.

✅ Question 1: What is the output of the above code? Explain why the values of a and ref are the same.

Part 1.2 Using References in Functions

References are often used in function parameters to allow functions to modify the arguments passed to them.

#include <iostream>

void swap(int& x, int& y) {
    int temp = x;
    x = y;
    y = temp;
}

int main() {
    int a = 5;
    int b = 10;
    swap(a, b);
    std::cout << "a: " << a << ", b: " << b << std::endl;

    return 0;
}

In this example, swap uses references to exchange the values of a and b.

✅ Question 2: How does the swap function modify the values of a and b? What would happen if the parameters were not references?

Part 1.3 Code Tracing and Conceptual Questions

#include <iostream>

void addOne(int& ref) {
    ref += 1;
}

int main() {
    int num = 5;
    addOne(num);
    std::cout << "num: " << num << std::endl;

    return 0;
}

✅ Question 3: Trace the execution of the above code. What is the value of num after the addOne function is called? Why?

Conceptual Questions

✅ Question 4: Can a reference be null? Explain why or why not.

✅ Question 5: Why might you use a reference parameter instead of a pointer parameter in a function?

✅ Question 6: What are the advantages and disadvantages of using references over pointers?

Part 1.4 Problem Set

✅ Program 1: Write a function that takes two references to integers and returns their product. The result should be stored in a third reference parameter.

✅ Program 2: Write a function that takes a reference to an integer and a reference to a string. If the integer is even, set the string to "Even". If the integer is odd, set the string to "Odd".

✅ Program 3: Write a function that swaps the values of two double variables using references. Test your function with different values.

✅ Program 4: Write a program that uses a reference to modify the value of a variable declared in the main function. Ensure that the changes are reflected in the main function.

Part 2. The Call Stack

In computer science, a call stack is a stack data structure that stores information about the active subroutines of a computer program. This kind of stack is also known as an execution stack, program stack, control stack, run-time stack, or machine stack, and is often shortened to just "the stack". Although maintenance of the call stack is important for the proper functioning of most software, the details are normally hidden and automatic in high-level programming languages.

2.1 What is the Call Stack?

The Call Stack, or more commonly just The Stack, can be thought of as a deck of notecards. Whenever a function begins, it creates a new card to be put on top of the stack of existing cards (hence the name).

We call these notecards stack frames. A stack frame stores data for a function call, essentially local variables related to that function.

In each stack frame the function allocates a small space for every variable it will need, along with any parameters it is given, and a space for whatever the function will return.

For a short visualization of how the stack works, please step through PythonTutor. The Python Tutor is a great resource if you're having trouble visualizing exactly what's happening in your code, and it's free! Make good use of it.

2.2 Getting started with the stack

Begin by compiling and running the starter code below with no additional flags (no output, or debugging quite yet).

#include <iostream>
int plus_one(int x) {
    return x + 1;
}
int plus_two(int x) {
    return plus_one(x + 1);
}
int main() {
    int result = 0;
    result = plus_one(0);
    result = plus_two(result);
    std::cout << result;
}

✅ Question 7: Examine what was printed out to the console. Where did this value come from? Describe how you think the call stack processed each operation (what was the order in which values were added to the stack?)

Let's test our theory by running this same code in PythonTutor.

✅ Question 8: Were you surprised by the activity in the stack on pythonTutor? What was different from your original guess?

Part 3. Functions [50 minutes]

3.1 What is a function?

A function is simply a block of statements that operates on certain parameters, and returns a value(though this last part isn't always true). In most modern high level languages functions serve as means of wrapping together repetitive code that you may use several times in your programs functionality. The syntax for a function in C++ is as follows:

<return type> <function name>(<parameters>) {
    <statements>
    <return statement> //again this is optional for a certain type of function(you'll see this later)
}

All of you have seen and used a function already this semester possibly without realizing. The main() function is a special function in the C++ language and holds all of the statements to be run when the program is called. Lets analyze an example of the main function.

#include <iostream>

int main() {
    std::cout << "Hello World!" << std::endl;
    return 0;
}

✅ Question 9. What is the return type of the main function?

✅ Question 10. What are the parameters to the main function?

As an example lets think about something you've been asked to do already this semester, testing if a number is odd / even. Code like this is something you might like to have when developing a program revolving around odd or even numbers and as a result writing out the statements everytime you want to test a number for being even. So lets say for example, I want to write a program that will endlessly prompt the user for input and output whether or not it is even(terminating if the user enters 0). My code might look something like this:

#include <iostream>
int main() {
    int input;
    while (1) {
        std::cout << "Enter a number: " << std::endl;
        std::cin >> input;

        if (!input) break;
        else {
            if (!(input % 2)) std::cout << "Even" << std::endl;
            else std::cout << "Odd" << std::endl;
        }
    }
    return 0;
}

We could also instead opt to move our functionailty for even number testing into a function, in which case we would have code like this:

#include <iostream>

bool is_even(int n) {
    if (!(n % 2)) return true;
    else return false;
}

int main() {
    int input;
    while (1) {
        std::cout << "Enter a number: " << std::endl;
        std::cin >> input;

        if (!input) break;
        else {
            if (is_even(input)) std::cout << "Even" << std::endl;
            else std::cout << "Odd" << std::endl;
        }
    }
    return 0;
}

As you can see this makes our code much more readable, as anyone who reads this program will know that is_even() tests a number for being even, even if they have no idea what the % operator does. We could even extrapolate this further and reduce our main function to a single line.

#include <iostream>

bool is_even(int n) {
    if (!(n % 2)) return true;
    else return false;
}

void run() {
    int input;
    while (1) {
        std::cout << "Enter a number: " << std::endl;
        std::cin >> input;

        if (!input) break;
        else {
            if (is_even(input)) std::cout << "Even" << std::endl;
            else std::cout << "Odd" << std::endl;
        }
    }
}

int main() {
    run();
    return 0;
}

✅ Question 11. What do you think it means for a fuction to return void?

✅ Question 12. Do you think that void functions can use return statements?

✅ Question 13. Copy the following code into your IDE, it should throw a syntax error at compile time. What does this error mean to you?

#include <iostream>

int main() {
    run();
    return 0;
}

bool is_even(int n) {
    if (!(n % 2)) return true;
    else return false;
}

void run() {
    int input;
    while (1) {
        std::cout << "Enter a number: " << std::endl;
        std::cin >> input;

        if (!input) break;
        else {
            if (is_even(input)) std::cout << "Even" << std::endl;
            else std::cout << "Odd" << std::endl;
        }
    }
}

3.2 ORDER MATTERS

You should have seen an error along the lines of error: 'run' was not declared in this scope. Why is this? Well its pretty self explanatory, when the main function goes to make a call to run(), it hasn't been declared yet. But we defined run() below main() why is it not declared? Well that is where the section title comes in, ORDER MATTERS. Since main() came before any declaration of run(), and since C++ reads from top to bottom the line that declares and defines run as function hasn't been executed before main tries to use it. What does this mean for us as programmers? Well it means we need to either declare all functions before main() or we need follow a strict hierarchy of needs between functions. That is rather verbose, so assume the following psuedo code layout has some fully functional functions in it for basic mathematic operations (addition, multiplication, exponents) if we want to use our functions without declaring them our structure might look something like this:

int add(int a, int b) {
    //add a to b
    //return result
}

int mult(int a, int b) {
    //add a to a sum, b times
    //return sum
}

int exp(int a, int b) {
    //multiply a times a b times
    //return result
}

int main() {
    std::cout << exp(2, 6) << std::endl;
}

This code shows the "strict hierarchy of needs between functions" mentioned earlier. Since exponentiation is repeated multiplication and multiplication is repeated addition, our code has to declare and define addition first, the multiplication,then exponentation. This may seem fine for this example, but this method of handling function dependancies is considerably cumbersome on just about anything that isn't basic math operations.

The second and far more common (as well as far more friendly) is to declare any functions that the program will have ahead of main, in which case our code might look like this:

int add(int a, int b);
int mult(int a, int b);
int exp(int a, int b);

int main() {
    std::cout << exp(2, 6) << std::endl;
}

int exp(int a, int b) {
    //multiply a times a b times
    //return result
}

int mult(int a, int b) {
    //add a to a sum, b times
    //return sum
}

int add(int a, int b) {
    //add a to b
    //return result
}

Note that C++ allows you to declare a function before it is defined, just like you've done with variables in the past. Following this structure allows you to write the functions in whatever order you want. These declarations of functions are called function headers and should at least include the return type of the function, and the types of its parameters. Meaning we could write the header for add() as either int add(int a, int b) or int add(int, int)

✅ Program 5. Write three functions in C++ to implement simple addition, multiplication, and exponentiation as with the function headers seen above and test them. Each function should use your base math functions (that is to say that mult() should us add(), exp() should use mult(), etc.)

3.3 Parameter Passing

C++ offers two ways to pass parameters to functions passing by value and passing by reference. Passing by value is the default in most languages and makes a copy of the data provided in that parameter and stores it as a local variable. Whereas passing by reference uses an alias for the variable provided in the argument. That is to say it makes another name that can refer to the same variable outside the scope of the function. The following is a code example of this concept:

#include <iostream>
void mystery(int& b, int c, int& a) {
   a ++;
   b --;
   c += a;
}

int main() {
   int a = 5;
   int b = 10;
   int c = 15;
   mystery(c, a, b);
   std::cout << a << ' ' << b << ' ' << c << '\n';
   return 0;
}

✅ Question 14. What is the output of the above code?

3.4 Problem Sets

✅ Program 6. Write a function that accepts 3 integers: a, b, and c. If a is even, return the sum of all ints between b and c. Otherwise, return the product of all ints between b and c.

✅ Program 7. Write a function that accepts a decimal number and returns that number in Binary.

✅ Program 8. Write a function that accepts an integer n and returns the # of digits. Ex. 1234 returns 4.

✅ Program 9. Write a function that accepts an integer n and returns the reverse of the integer. Ex. 1234 returns 4321.

Hint: Use your solution from Program 9 to help you solve this!

Part 4. Submission [5 minutes]

Each group will submit a single .zip file named lab-04.zip containing all your answers to the lab questions in your lab-04.txt and all of your .cpp source code files on Gradescope before the end of your lab section. All submissions should be made by a group/team. Individual submissions will not be accepted. Instructions to download your lab-04.txt file can be found in the IDE introduction page that you read in lab-01. For your convenience, that page is relinked here.