Lab 2 -- Functions and Floating Point Equality
CPE101
Winter 2008
Updates and Corrections
Due Date:
By the end of your last lab of week 2 (unless your instructor tells you otherwise).
Objectives
Overview
In this lab you will be writing a few functions and developing tests to verify their correctness.
Functions are simply named chunks of code that can be called (by name) anywhere in your program. Calling a function affects the flow-of-execution of your program. When you call a function the next line of code to execute, roughly speaking is the first line of code in the called function. When the function is done, execution returns to the line of code just after the original call. That "detour" can involve any number of other function calls and that, along with selection statements (if-else) and loops (while, for, et cetera) makes up much of the expressive power of a high-level language.
You have already called a function in your first lab, specifically the printf function. Hopefully you were curious enough about your first program to try to figure out how it worked and correctly conjectured that the printf function was what caused the greeting to display. This function is one of many found in the C Standard Library and, like most, it performs an operation that is commonly needed in many programs. The printf function also demonstrates a few important characteristics to "good" function design. First, it has a meaningful name, that is, the name gives you a good idea about what it does and, second, it separates the data of the operation (known as function arguments and/or parameters) from the action (code or logic) of the operation. Notice that you "pass" a string to printf, in our case "Hello World!" and that is what is displayed. You can pass any string to printf and it will print that specific string. This is often called procedural abstraction and is a cornerstone of good function design. Note that printf's primary operation is to have affect on the outside world, i.e., print to the console but that many (most?) functions return some value or result.
So, when designing functions you must do several things:
1. Determine what the function should do. A good function almost always does something that could be considered a single task.
2. Come up with a good name for the function. Good names should be short and to the point and be obvious enough that almost anyone can figure out what the function does with a minimum of thought.
3. Determine the function's inputs (arguments or parameters), if any.
4. Determine the function's the return value and type, if any.
Example
So, let's suppose you want to write a little program that prompts a user for two integer values and reports which is larger. You could write the entire solution in the required main function as follows:
#include <stdio.h>
int main()
{
int x, y;
printf("Enter an integer: ");
scanf("%d", &x);
printf("Enter an integer: ");
scanf("%d", &y);
if (x < y)
{
printf("The larger value is %d\n", y);
}
else
{
printf("The larger value is %d\n", x);
}
return 0;
}
It doesn't look too bad, but what if you wanted to have the user enter 100 values and report the largest value? With the approach above you would need 100 int variables and you would have to duplicate the printf-scanf calls 100 times, and the if-else logic would be, well, lets just say very long and very error-prone. Let's see if a little procedural abstraction, i.e., functions can help.
#include <stdio.h>
/* Function prototypes */
int getInt();
int max(int x, int y);
int main()
{
int newValue, maxValue;
maxValue = getInt();
newValue = getInt();
maxValue = max(maxValue, newValue);
printf("The larger value is %d\n", maxValue);
return 0;
}
int getInt()
{
int x;
printf("Enter an integer: ");
scanf("%d", &x);
return x;
}
int max(int a, int b)
{
if (a > b )
{
return a;
}
else
{
return b;
}
}
Yes, this one is longer for two values, but would be much shorter for 100 values. Less obvious is that it would be much easier to debug and fix. For example, if there was an error in the max logic it would be isolated to the small and simple max function rather than a very long series of if-else statements.
Notice the prototypes near the top of the source file -- they are necessary if you want to call a function in your code before you actually write it. In compiler-ese this is because the function must be declared before it is used -- the prototype tells the compiler to chill -- and to expect to see a function matching the prototype before it is through (if it doesn't the link phase will fail). An alternative to prototyping would be to put any function that calls another after the one it calls but this doesn't work if you have two functions that call each other -- they can't both be after the other!
Note that prototypes must have the same return type, function name, and parameter type and order as the actual function implementation. Notice too that the parameter names in the prototype do not have to match the names in the actual function implementation -- in fact, you don't even need parameter names in the function prototypes. It is generally a good idea to include them for the sake of documentation -- so make the names meaningful!
Lab Requirements (Or the answer to "So what am I supposed to do?")
#include <stdio.h>
int main()
{
double a = 0.2;
double b = 0.3;
printf("a * b == %f\n", a * b);
printf("a * b with more information displayed %.32f\n", a * b);
return 0;
}
Did you guess correctly? As you can see, floating point numbers are not always exactly correct. If you want to learn more about why take a look at this wikipedia discussion of floating point numbers.
Because of this (and other) behavior of floating point numbers it is vitally important to be careful when checking them for equality. It is generally good advice to not use the built in equality operator (==) but instead to use some kind of logic to determine approximate equality. The degree of precision required is application dependent and only the application programmer knows how precise to be. A common approach is to write a function that takes three double (or float) parameters, two being the values to compare for equality, and a third representing an acceptable +/- range (often called epsilon) to still be considered equal. If the difference of the values is within that range the function would return true, otherwise false.
Remember that in C, we use the int data type to represent boolean values, 0 is false, not zero is true (often, but not always 1).
Be careful when determining the difference of the two values there may be a sign (positive/negative) issue when the numbers are not exactly equal -- your solution should work in all cases. Hunt around in Standard Library for C (hint: math.h and a web search might be the best/quickest way to find out what function(s) might be helpful here).
After doing your homework, design and implement such a function and use it in a program called part3.c that allows someone to enter two real numbers and an epsilon value (also a real number) and reports if the two numbers are equal (+/- epsilon) or not. Give the function and parameters meaningful names. The function should not do any user I/O -- the user I/O should be done in main only. Test your program well and, if your instructor is doing demonstrations, demonstrate your lab when you think it is well implemented and tested.
Handing in Your Source Electronically . . .
11:59 vogon ~$ handin mhaungs Lab2-yy part1.c part2.c part3.c