Exercises -- Intro To C++ Programming

Total Page:16

File Type:pdf, Size:1020Kb

Exercises -- Intro To C++ Programming

Exercises for Introduction to C++ Programming Page 1 / 54

Exercises for Intro to C++ Programming

Index

Last updated: May 1, 2013 Exercises for Introduction to C++ Programming Page 2 / 54

Chap 01 - C++ Basics

Ex 1.01

Fully parenthesize the following expressions, in order to show the precedence and associativity:

1 + 2 + 3 + 4 16 * 4 / 2 * 3 a > b ? a: c > d ? e: f a & b && c & d

Ex 1.02

Without running it, can you work out what value the following code is going to output, and why?

unsigned s = 555; int i = (s >> 4) & ~(~0 << 3); cout << i;

Ex 1.03 (even vs. odd)

An even number is one which when divided by 2 has no remainder. An even number “is evenly divisible by 2.”

Write a program that asks the user for an integer and then tells if it is even or odd.

Is there a more efficient way to determine whether an integer is even? More efficient than doing the division and checking the remainder? Exercises for Introduction to C++ Programming Page 3 / 54

Chap 02 - Flow of Control

Ex 2.01

Rewrite the following loops as for loops. Place both versions into a program and check to see that both versions do, in fact, produce identical outputs. a. int i = 1; while (i <= 10) { if (i < 5 && i != 2) cout << 'X'; i++; } b. int i = 1; while (i <<= 10) { cout << 'X'; i = i + 3; } c. long n = 100; do { cout 6 < 'X'; n = n + 100; } while (n < 1000);

Ex 2.02

The following code is supposed to output every word in the file and only every word in the file, but it does not quite work. What is wrong?

bool moreToRead = true; while (moreToRead) { string text; moreToRead = inputStream >> text; cout << text << endl; } Exercises for Introduction to C++ Programming Page 4 / 54

After correcting the code, embed it in a program and check to see that it works correctly.

Ex 2.03 (Body weight)

A simple rule to estimate your ideal body weight is to allow 110 pounds for the first 5 feet of height and 5 pounds for each additional inch. Write a program with a variable for the height of a person in feet and another variable for the additional inches. Assume the person is at least 5 feet tall. For example, a person that is 6 feet and 3 inches tall would be represented with a variable that stores the number 6 and another variable that stores the number 3. Based on these values, calculate and output the ideal body weight.

Ex 2.04 (Babylonian algorithm for square root)

The Babylonian algorithm to compute the square root of a positive number n is as follows:

1. Make a guess at the answer (you can pick n / 2 as your initial guess) 2. Compute r = n / guess 3. Set guess = (guess + r) / 2 4. Go back to step 2 until the last two guess values are within 1% of each other

Write a program that inputs a double for n, iterates through the Babylonian algorithm until the guess is within 1% of the previous guess, and outputs the answer as a double to two decimal places. Your answer should be accurate even for large values of n.

Ex 2.05

You have just purchased a stereo system that costs $1,000 on the following credit plan: no down payment, an interest rate of 18% per year (and hence 1.5% per month), and monthly payments of $50. The monthly payment of $50 is used to pay the interest, and whatever is left is used to pay part of the remaining debt. Hence, the first month you pay 1.5% of $1,000 in interest. That is $15 in interest. So, the remaining $35 is deducted from your debt, which leaves you with a debt of $965.00. The next month, you pay interest of 1.5% of $965.00, which is $14.48. Hence, you can deduct $35.52 (which is $50 – $14.48) from the amount you owe.

Write a program that tells you how many months it will take you to pay off the loan, as well as the total amount of interest paid over the life of the loan. Use a loop to calculate the amount of interest and the size of the debt after each month. (Your final program need not output the monthly amount of interest Exercises for Introduction to C++ Programming Page 5 / 54 paid and remaining debt, but you may want to write a preliminary version of the program that does output these values.) Use a variable to count the number of loop iterations and hence, the number of months until the debt is zero. You may want to use other variables as well. The last payment may be less than $50 if the debt is small, but do not forget the interest. If you owe $50, your monthly payment of $50 will not pay off your debt, although it will come close. One month’s interest on $50 is only 75 cents.

Ex 2.06 (Fibonacci numbers)

The Fibonacci numbers Fn are defined as follows: F0 is 0, F1 is 1, and Fi+2 = Fi+1 + Fi, for i = 0, 1, 2,… In other words, each number is the sum of the previous two numbers. The first few Fibonacci numbers are 0, 1, 1, 2, 3, 5, and 8.

One place where these numbers occur is as certain population growth rates. If a population has no deaths, then the series shows the size of the population after each time period. It takes an organism two time periods to mature to reproducing age, and then the organism reproduces once every time period. The formula applies most straightforwardly to asexual reproduction at a rate of one offspring per time period. In any event, the green crud population grows at this rate and has a time period of five days. Hence, if a green crud population starts out as 10 pounds of crud, then in 5 days, there is still 10 pounds of crud; in 10 days, there is 20 pounds of crud; in 15 days, 30 pounds; in 20 days, 50 pounds; and so forth.

Write a program that takes both the initial size of a green crud population (in pounds) and a number of days as input and outputs the number of pounds of green crud after that many days. Assume that the population size is the same for four days and then increases every fifth day. Your program should allow the user to repeat this calculation as often as desired.

Ex 2.07 (Fibonacci experiment)

An interesting experiment: Calculate the ratio of Fi+1 / Fi over the range 1..20. See if you find a “pattern” in the results.

Ex 2.08 (Cryptarithmetic puzzles)

In cryptarithmetic puzzles, mathematical equations are written using letters. Each letter can be a digit from 0 to 9, but no two letters can be the same. Here is a sample problem: SEND + MORE = MONEY A solution to the puzzle is S = 9, R = 8, O = 0, M = 1, Y = 2, E = 5, N = 6, D = 7. Exercises for Introduction to C++ Programming Page 6 / 54

Write a program that finds a solution to the cryptarithmetic puzzle of the following: TOO + TOO + TOO + TOO = GOOD.

The simplest technique is to use a nested loop for each unique letter (in this case T, O, G, D). The loops would systematically assign the digits from 0 to 9 to each letter. For example, it might first try T = 0, O = 0, G = 0, D = 0, then T = 0, O = 0, G = 0, D = 1, then T = 0, O = 0, G = 0, D = 2, etc., up to T = 9, O = 9, G = 9, D = 9. In the loop body, test that each variable is unique and that the equation is satisfied. Output all the values for the letters that satisfy the equation. Exercises for Introduction to C++ Programming Page 7 / 54

Chap 03 - Function Basics

Ex 3.01 (Babylonian square root as a function)

Recast your solution to Ex 2.04 in the form of a function. Use 0.1% as the limiting condition. Display all numbers with 4 digits to the right of the decimal point. Build a driving routine that compares the output of your function to the built-in sqrt() function over the range 0..10, in steps of 0.1. Display your results as a 3-column table.

Ex 3.02 (Gauge inflation)

Write a program to gauge the rate of inflation for the past year. The program asks for the price of an item (such as a hot dog or a one-carat diamond) both one year ago and today. It estimates the inflation rate as the difference in price divided by the year-ago price. Your program should allow the user to repeat this calculation as often as the user wishes. Define a function to compute the rate of inflation. The inflation rate should be a value of type double giving the rate as a percentage, for example 5.3 for 5.3%.

Ex 3.03

Enhance your program from the Ex 3.02 by having it also display the estimated price of the item in one and in two years from the time of the calculation. The increase in cost over one year is estimated as the inflation rate times the price at the start of the year. Define a second function to determine the estimated cost of an item in a specified number.

Ex 3.04 (The game of craps)

In the game of craps, a “Pass Line” bet proceeds as follows. The first roll of the two, six-sided dice in a craps round is called the “come out roll.” The bet immediately wins when the come out roll is 7 or 11, and loses when the come out roll is 2, 3, or 12. If 4, 5, 6, 8, 9, or 10 is rolled on the come out roll, that number becomes “the point.” The player keeps rolling the dice until either 7 or the point is rolled. If the point is rolled first, then the player wins the bet. If the player rolls a 7 first, then the player loses.

Write a program that plays craps using these rules so that it simulates a game without human input. Instead of asking for a wager, the program should Exercises for Introduction to C++ Programming Page 8 / 54 calculate whether the player would win or lose. Create a function that simulates rolling the two dice and returns the sum. Add a loop so that the program plays 10,000 games. Add counters that count how many times the player wins, and how many times the player loses. At the end of the 10,000 games, compute the probability of winning, as Wins / (Wins + Losses), and output this value. Over the long run, who is going to win more games of craps, you or the house?

Ex 3.05 (Roman numeral conversion)

The Romans did not use the same number system that we do today (Arabic). In the Roman system the following characters were used to represent numbers: I = 1 V = 5 X = 10 L = 50 C = 100 M = 1000

For simple numbers, the numerals are placed in “descending order.” So, for example, here are some simple Roman numbers and their decimal values:

VI = 6 XII = 12 XXXII = 32 MCLXVI = 1,166

But things actually are a bit more complicated. If you place a lower value numeral before a higher one, then the value of the lower numeral is subtracted from the higher one. Only a single lower numeral may be placed in this manner. For example:

IV = 4 XL = 40 CM = 900

MCMXLIV = 1,944

Write a program that accepts instructions to either convert from Roman to Decimal or Decimal to Roman, and the value to be converted. Exercises for Introduction to C++ Programming Page 9 / 54

Chap 04 - Parameters + Overloading

Ex 4.01 (Area of a triangle)

The area of an arbitrary triangle can be computed using the formula

Area = [ s(s – a)(s – b)(s – c) ] ^ 1/2 where a, b, and c are the lengths of the sides, and s is the semiperimeter: s = (a + b + c) / 2

Write a void function that uses five parameters: three value parameters that provide the lengths of the edges, and two reference parameters that compute the area and perimeter (not the semiperimeter). Make your function robust. Note that not all combinations of a, b, and c produce a triangle. Your function should produce correct results for legal data and reasonable results for illegal combinations.

Ex 4.02 (Roman numeral recast)

“Repackage” your program from Ex 3.05 so that there are (at least) two functions – one for converting in each direction – and a driving routine.

Ex 4.03

In the land of Puzzlevania, Aaron, Bob, and Charlie had an argument over which one of them was the greatest puzzle-solver of all time. To end the argument once and for all, they agreed on a duel to the death. Aaron was a poor shot and only hit his target with a probability of 1/3. Bob was a bit better and hit his target with a probability of 1/2. Charlie was an expert marksman and never missed. A hit means a kill and the person hit drops out of the duel.

To compensate for the inequities in their marksmanship skills, the three decided that they would fire in turns, starting with Aaron, followed by Bob, and then by Charlie. The cycle would repeat until there was one man standing. That man would be remembered for all time as the Greatest Puzzle-Solver of All Time. An obvious and reasonable strategy is for each man to shoot at the most accurate shooter still alive, on the grounds that this shooter is the deadliest and has the best chance of hitting back.

Write a program to simulate the duel using this strategy. Your program should use random numbers and the probabilities given in the problem to determine Exercises for Introduction to C++ Programming Page 10 / 54 whether a shooter hits his target. You will likely want to create multiple subroutines and functions to complete the problem. Once you can simulate a duel, add a loop to your program that simulates 10,000 duels. Count the number of times that each contestant wins and print the probability of winning for each contestant (e.g., for Aaron your program might output “Aaron won 3595/10,000 duels or 35.95%”). An alternate strategy is for Aaron to intentionally miss on his first shot. Modify the program to accommodate this new strategy and output the probability of winning for each contestant. What strategy is better for Aaron, to intentionally miss on the first shot or to try and hit the best shooter?

Ex 4.04 (Making change)

Write a program that tells what coins to give out for any amount of change from 1 cent to 99 cents. For example, if the amount is 86 cents, the output would be something like the following:

86 cents can be given as 3 quarter(s) 1 dime(s) and 1 penny(pennies)

Use coin denominations of 25 cents (quarters), 10 cents (dimes), and 1 cent (pennies). Do not use nickel and half-dollar coins. Your program will use the following function (among others):

// Precondition: 0 < coinValue < 100; 0 <= amountLeft < 100. // Postcondition: number has been set equal to the maximum number // of coins of denomination coinValue cents that can be obtained // from amountLeft cents. amountLeft has been decreased by the // value of the coins, that is, decreased by number*coinValue. void computeCoin(int coinValue, int& number, int& amountLeft);

For example, suppose the value of the variable amountLeft is 86. Then, after the following call, the value of number will be 3 and the value of amountLeft will be 11 (because if you take three quarters from 86 cents, that leaves 11 cents):

computeCoins(25, number, amountLeft);

Include a loop that lets the user repeat this computation for new input values until the user says he or she wants to end the program. (Hint: Use integer division and the % operator to implement this function.)

Ex 4.05 (Eight queens problem)

The “Eight queens problem” is to use an ordinary chess board and place 8 queens on the board in such a manner so that no queen is under attack. Exercises for Introduction to C++ Programming Page 11 / 54

Write a program that finds one solution to this problem. Consider how you might modify your program so that it finds all solutions. Exercises for Introduction to C++ Programming Page 12 / 54

Chap 05 - Arrays

Ex 5.01

Suppose list is an array of five components of type int. What is stored in list after the following C++ code executes?

for (int i = 0; i < 5; i++) { list[i] = 2 * i + 5; if (i % 2 == 0) list[i] = list[i] - 3; }

Ex 5.02 (Build a sort routine)

Using the algorithm of your choice – selection sort, insertion sort, bubble sort, whatever – write a function that accepts an array of type int and n, the number of entries in the array. The function short sort the array into ascending order

Build a driver that creates an array of 25 random int value, all values in the range 1..25 (with possible duplicated values). It should display the array. Then it should call your sort routine. Finally it should display the resulting sorted array.

Ex 5.03 (Search a sorted array)

After completing Ex 5.01 you have a function that will sort an array. Now, build a (search) function that accepts two arguments – a sorted array of int’s and an int value, n. The function must return the index of the first instance of n in the array if it is present. If n is not present in the array, return -1.

Build a driver to test your function.

If your search function is allowed to modify the contents of the array, can it be made more efficient? If so, how?

Ex 5.04 Exercises for Introduction to C++ Programming Page 13 / 54

Write a program that reads in an array of type int. You may assume that there are fewer than 50 entries in the array. Your program determines how many entries are used. The output is to be a two-column list. The first column is a list of the distinct array elements; the second column is the count of the number of occurrences of each element. The list should be sorted on entries in the first column, largest to smallest. For the array values:

–12 3 –12 4 1 1 –12 1 –1 1 2 3 4 2 3 –12 the output should be

N Count 4 2 3 3 2 2 1 4 –1 1 –12 4

Ex 5.05 (Multiple precision computation)

An array can be used to store large integers one digit at a time. For example, the integer 1234 could be stored in the array a by setting a[0] to 1, a[1] to 2, a[2] to 3, and a[3] to 4. However, for this exercise you might find it more useful to store the digits backward, that is, place 4 in a[0], 3 in a[1], 2 in a[2], and 1 in a[3].

In this exercise you will write a program that reads in two positive integers that are 20 or fewer digits in length and then outputs the sum of the two numbers. Your program will read the digits as values of type char so that the number 1234 is read as the four characters '1', '2', '3', and '4'. After they are read into the program, the characters are changed to values of type int. The digits will be read into a partially filled array, and you might find it useful to reverse the order of the elements in the array after the array is filled with data from the keyboard. (Whether or not you reverse the order of the elements in the array is up to you. It can be done either way, and each way has its advantages and disadvantages.)

Your program will perform the addition by implementing the usual paper-and- pencil addition algorithm. The result of the addition is stored in an array of size 20 and the result is then written to the screen. If the result of the addition is an integer with more than the maximum number of digits (that is, more than 20 digits), then your program should issue a message saying that it has encountered “integer overflow.” You should be able to change the maximum length of the integers by changing only one globally defined constant. Include a loop that allows the user to continue to do more additions until the user says the program should end. Exercises for Introduction to C++ Programming Page 14 / 54

Of course, your multiple precision addition code should be formulated as a function that accepts a pair of int arrays along with n, the number of digits that are permitted.

Ex 5.06 (Birthday paradox)

The birthday paradox is that there is a surprisingly high probability that two people in the same room happen to share the same birthday. By birthday, we mean the same day of the year (ignoring leap years), but not the exact birthday including the birth year or time of day.

Write a program that approximates the probability that two people in the same room have the same birthday, for 2 to 50 people in the room. The program should use simulation to approximate the answer. Over many trials (say, 5000), randomly assign birthdays to everyone in the room. Count up the number of times at least two people have the same birthday, and then divide by the number of trials to get an estimated probability that two people share the same birthday for a given room size. Your output should look something like the following. It will not be exactly the same due to the random numbers involved:

For 2 people, the probability of two birthdays is about 0.002 For 3 people, the probability of two birthdays is about 0.0082 For 4 people, the probability of two birthdays is about 0.0163 ... For 49 people, the probability of two birthdays is about 0.9654 For 50 people, the probability of two birthdays is about 0.969

Ex 5.07 (The game of “Life”)

The mathematician John Horton Conway invented the “Game of Life.” Though not a “game” in any traditional sense, it provides interesting behavior that is specified with only a few rules. This exercise asks you to write a program that allows you to specify an initial configuration. The program follows the rules of Life (listed shortly) to show the continuing behavior of the configuration.

LIFE is an organism that lives in a discrete, two-dimensional world. While this world is actually unlimited, we do not have that luxury, so we restrict the array to 80 characters wide by 22 character positions high. If you have access to a larger screen, by all means use it. This world is an array with each cell capable of holding one LIFE cell. Generations mark the passing of time. Each generation brings births and deaths to the LIFE community. The births and deaths follow this set of rules: Exercises for Introduction to C++ Programming Page 15 / 54

1. We define each cell to have eight neighbor cells. The neighbors of a cell are the cells directly above, below, to the right, to the left, diagonally above to the right and left, and diagonally below, to the right and left. 2. If an occupied cell has zero or one neighbor, it dies of loneliness. If an occupied cell has more than three neighbors, it dies of overcrowding. 3. If an empty cell has exactly three occupied neighbor cells, there is a birth of a new cell to replace the empty cell. 4. Births and deaths are instantaneous and occur at the changes of generation. A cell dying for whatever reason may help cause birth, but a newborn cell cannot resurrect a cell that is dying, nor will a cell’s death prevent the death of another, say, by reducing the local population. * Examples: *** becomes * then becomes *** again, and so on. * Notes: Some configurations grow from relatively small starting configurations. Others move across the region. It is recommended that for text output you use a rectangular char array with 80 columns and 22 rows to store the LIFE world’s successive generations. Use an * to indicate a living cell and use a blank to indicate an empty (or dead) cell. If you have a screen with more rows than that, by all means make use of the whole screen.

Suggestions: Look for stable configurations. That is, look for communities that repeat patterns continually. The number of configurations in the repetition is called the period. There are configurations that are fixed, that is, they continue without change. A possible project is to find such configurations.

Hints: Define a void function named generation that takes the array we call world, an 80-column by 22-row array of type char, which contains the initial configuration. The function scans the array and modifies the cells, marking the cells with births and deaths in accord with the rules listed previously. This involves examining each cell in turn and either killing the cell, letting it live, or, if the cell is empty, deciding whether a cell should be born. There should be a function display that accepts the array world and displays the array on the screen. Some sort of time delay is appropriate between calls to generation and display. To do this, your program should generate and display the next generation when you press Return. You are at liberty to automate this, but automation is not necessary for the program.

Ex 5.08 (Encryption)

Traditional password entry schemes are susceptible to “shoulder surfing” in which an attacker watches an unsuspecting user enter their password or PIN number and uses it later to gain access to the account. One way to combat this problem is with a randomized challenge-response system. In these systems, the user enters different information every time based on a secret in response to a randomly generated challenge. Consider the following scheme in which the password consists of a five-digit PIN number (00000 to 99999). Exercises for Introduction to C++ Programming Page 16 / 54

Each digit is assigned a random number that is 1, 2, or 3. The user enters the random numbers that correspond to their PIN instead of their actual PIN numbers.

For example, consider an actual PIN number of 12345. To authenticate, the user would be presented with a screen such as

PIN: 0 1 2 3 4 5 6 7 8 9 NUM: 3 2 3 1 1 3 2 2 1 3

The user would enter 23113 instead of 12345. This does not divulge the password even if an attacker intercepts the entry because 23113 could correspond to other PIN numbers, such as 69440 or 70439. The next time the user logs in, a different sequence of random numbers would be generated, such as

PIN: 0 1 2 3 4 5 6 7 8 9 NUM: 1 1 2 3 1 2 2 3 3 3

Your program should simulate the authentication process. Store an actual PIN number in your program. The program should use an array to assign random numbers to the digits from 0 to 9. Output the random digits to the screen, input the response from the user, and output whether or not the user’s response correctly matches the PIN number. Exercises for Introduction to C++ Programming Page 17 / 54

Chap 06 - Structures + Classes

Ex 6.01 (Fraction class)

Define a class called Fraction. This class is used to represent a ratio of two integers. Include mutator methods that allow the user to set the numerator and the denominator. Also include a method that returns the value of numerator divided by denominator as a double. Include a method that displays the fraction in the form #/# where both #’s are integers. E.g., 3/4. If the denominator is 1, only display the numerator. Internally in the class maintain the fraction in “reduced form.” That means that the numerator and denominator have no common factors. This will require finding the greatest common divisor for the numerator and denominator, then dividing both by that number. The gcd() function should be internal to your class.

Embed your class in a test program that accepts input from the console.

Ex 6.02 (Complex number class)

Create a class called Complex that represents complex numbers. Remember that a Complex number is composed of two parts – a real part and an imaginary part. C = A + Bi, where i = sqrt(-1).

Populate your class with appropriate constructors. Define functions for Complex addition, subtractions, multiplication, and division. You’ll need to include functions that deal with the input and output of Complex numbers.

Build a driver that repeatedly requests input from the console – two complex values and an operation to be performed.

Ex 6.03 (Pizzas)

Create a class named Pizza that stores information about a single pizza. It should contain the following:

• Private instance variables to store the size of the pizza (small, medium, or large), the number of cheese toppings, the number of pepperoni toppings, and the number of ham toppings • Constructor(s) that set all of the instance variables • Public methods to get and set the instance variables. • A public method named calcCost() that returns a double that is the cost of the pizza. Pizza cost is determined by: Exercises for Introduction to C++ Programming Page 18 / 54

o Small: $10 + $2 per topping o Medium: $12 + $2 per topping o Large: $14 + $2 per topping • A public method named getDescription() that returns a String containing the pizza size, quantity of each topping, and the pizza cost as calculated by calcCost().

Write test code to create several pizzas and output their descriptions. For example, a large pizza with one cheese, one pepperoni and two ham toppings should cost a total of $22.

Ex 6.04 (Extended Pizzas)

This exercise extends the previous one. Create a PizzaOrder class that allows up to three pizzas to be saved in an order. Each pizza saved should be a Pizza object as described in Ex 4.01. In addition to appropriate instance variables and constructors, add the following methods:

• public void setNumPizzas(int numPizzas) – sets the number of pizzas in the order. numPizzas must be between 1 and 3. • public void setPizza1(Pizza pizza1) – sets the first pizza in the order. • public void setPizza2(Pizza pizza2) – sets the second pizza in the order. • public void setPizza3(Pizza pizza3) – sets the third pizza in the order. • public double calcTotal() – returns the total cost of the order.

Write a main method to test the class. The setPizza2 and setPizza3 methods will be used only if there are two or three pizzas in the order, respectively. Sample code illustrating the methods is shown below. Note that first three lines are incomplete. You must complete them as part of the exercise.

Pizza pizza1 = // Code to create a large pizza, 1 cheese, 1 ham Pizza pizza2 = // Code to create a medium pizza, 2 cheese, 2 pepperoni PizzaOrder order = // Code to create an order setNumPizzas(2); // 2 pizzas in the order setPizza1(pizza1); // Set first pizza order setPizza2(pizza2); // Set second pizza double total = order calcTotal(); // Should be 18 + 20 = 38 Exercises for Introduction to C++ Programming Page 19 / 54

Chap 07 - Constructors + Other Tools

Ex 7.01 (Month)

Define a class called Month that is an abstract data type for a month. Your class will have one member variable of type int to represent a month (1 for January, 2 for February, and so forth). Include all the following member functions: a constructor to set the month using the first three letters in the name of the month as three arguments, a constructor to set the month using an integer as an argument (1 for January, 2 for February, and so forth), a default constructor, an input function that reads the month as an integer, an input function that reads the month as the first three letters in the name of the month, an output function that outputs the month as an integer, an output function that outputs the month as the first three letters in the name of the month, and a member function that returns the next month as a value of type Month. Embed your class definition in a test program.

Ex 7.02 (Little Red Counter)

My mother always took a little red counter to the grocery store. The counter was used to keep tally of the amount of money she would have spent so far on that visit to the store if she bought everything in the basket. The counter had a four-digit display, increment buttons for each digit, and a reset button. An overflow indicator came up red if more money was entered than the $99.99 it would register. (This was a long time ago.)

Write and implement the member functions of a class Counter that simulates and slightly generalizes the behavior of this grocery store counter. The constructor should create a Counter object that can count up to the constructor’s argument. That is, Counter(9999) should provide a counter that can count up to 9999. A newly constructed counter displays a reading of 0. The member function void reset( ); sets the counter’s number to 0. The member function void incr1( ); increments the units digits by 1, void incr10( ); increments the tens digit by 1, and void incr100( ); and void incr1000( ); increment the next two digits, respectively. Accounting for any carrying when you increment should require no further action than adding an appropriate number to the private data member. A member function bool overflow( ); detects overflow. (Overflow is the result of incrementing the counter’s private data member beyond the maximum entered at counter construction.) Use this class to provide a simulation of my mother’s little red clicker. Even though the display is an integer, in the simulation, the rightmost (lower order) two digits are always thought of as cents and tens of cents, the next digit is dollars, and the fourth digit is tens of dollars. Exercises for Introduction to C++ Programming Page 20 / 54

Provide keys for cents, dimes, dollars, and tens of dollars. Unfortunately, no choice of keys seems particularly mnemonic. One choice is to use the keys asdfo: a for cents, followed by a digit 1 to 9; s for dimes, followed by a digit 1 to 9; d for dollars, followed by a digit 1 to 9; and f for tens of dollars, again followed by a digit 1 to 9. Each entry (one of asdf followed by 1 to 9) is followed by pressing the Return key. Any overflow is reported after each operation. Overflow can be requested by pressing the o key.

Ex 7.03 (Hot Dogs)

You operate several hot dog stands distributed throughout town. Define a class named HotDogStand that has a member variable for the hot dog stand’s ID number and a member variable for how many hot dogs the stand sold that day. Create a constructor that allows a user of the class to initialize both values.

Also create a function named JustSold that increments the number of hot dogs the stand has sold by one. This function will be invoked each time the stand sells a hot dog so that you can track the total number of hot dogs sold by the stand. Add another function that returns the number of hot dogs sold.

Finally, add a static variable that tracks the total number of hot dogs sold by all hot dog stands and a static function that returns the value in this variable. Write a main function to test your class with at least three hot dog stands that each sell a variety of hot dogs.

Ex 7.04 (The Beautiful Princess)

In an ancient land, the beautiful princess Eve had many suitors. She decided on the following procedure to determine which suitor she would marry. First, all of the suitors would be lined up one after the other and assigned numbers. The first suitor would be number 1, the second number 2, and so on up to the last suitor, number n. Starting at the first suitor she would then count three suitors down the line (because of the three letters in her name) and the third suitor would be eliminated from winning her hand and removed from the line. Eve would then continue, counting three more suitors, and eliminating every third suitor. When she reached the end of the line she would continue counting from the beginning.

For example, if there were six suitors then the elimination process would proceed as follows:

123456 initial list of suitors, start counting from 1 12456 suitor 3 eliminated, continue counting from 4 1245 suitor 6 eliminated, continue counting from 1 Exercises for Introduction to C++ Programming Page 21 / 54

125 suitor 4 eliminated, continue counting from 5 15 suitor 2 eliminated, continue counting from 5 1 suitor 5 eliminated, 1 is the lucky winner

Write a program that uses a vector to determine which position you should stand in to marry the princess if there are n suitors. You will find the following function from the Vector class useful:

// Removes element at position iter v.erase(iter);

For example, to use this function to erase the fourth element from the beginning of a vector variable named theVector, use theVector.erase(theVector.begin( ) + 3);

The number 3 is used because the first element in the vector is at index position 0. (I.e., vector indices start at zero – they are “zero-based.”) Exercises for Introduction to C++ Programming Page 22 / 54

Chap 08 - Operator Overloading, Friends, + References

Ex 8.01 (Rational Numbers)

Define a class for rational numbers. A rational number is a number that can be represented as the quotient of two integers. For example, 1/2, 3/4, 64/2, and so forth are all rational numbers. (By 1/2 and so on we mean the everyday fraction, not the integer division this expression would produce in a C++ program.) Represent rational numbers as two values of type int, one for the numerator and one for the denominator. Call the class Rational. Include a constructor with two arguments that can be used to set the member variables of an object to any legitimate values. Also include a constructor that has only a single parameter of type int; call this single parameter wholeNumber and define the constructor so that the object will be initialized to the rational number wholeNumber /1. Include a default constructor that initializes an object to 0 (that is, to 0/1). Overload the input and output operators >> and <<. Numbers are to be input and output in the form 1/2, 15/32, 300/401, and so forth. Note that the numerator, the denominator, or both may contain a minus sign, so -1/2, 15/-32, and -300/-401 are also possible inputs. Overload all the following operators so that they correctly apply to the type Rational: ==, <, <=, >, >=, +, -, *, and /. Write a test program to test your class.

Hints: Two rational numbers a/b and c/d are equal if a*d equals c*b. If b and d are positive rational numbers, a/b is less than c/d provided a*d is less than c*b. You should include a function to normalize the values stored so that, after normalization, the denominator is positive and the numerator and denominator are as small as possible. For example, after normalization 4/-8 would be represented the same as -1/2.

Ex 8.02 (Complex Numbers)

Define a class for complex numbers. A complex number is a number of the form

a + b*i where for our purposes, a and b are numbers of type double, and i is a number that represents the quantity sqrt(-1). Represent a complex number as two values of type double. Name the member variables real and imaginary. (The variable for the number that is multiplied by i is the one called imaginary.) Call the class Complex. Include a constructor with two parameters of type double that can be used to set the member variables of an object to any values. Include a constructor that has only a single parameter of type Exercises for Introduction to C++ Programming Page 23 / 54 double; call this parameter realPart and define the constructor so that the object will be initialized to realPart + 0*i. Include a default constructor that initializes an object to 0 (that is, to 0 + 0*i ). Overload all the following operators so that they correctly apply to the type Complex: ==, +, -, *, >>, and <<. You should also write a test program to test your class. Hints: To add or subtract two complex numbers, add or subtract the two member variables of type double. The product of two complex numbers is given by the following formula:

(a + b*i)*(c + d*i) = = (a*c - b*d) + (a*d + b*c)*i

In the interface file, you should define a constant i as follows:

const Complex i(0, 1);

This defined constant i will be the same as the i discussed above.

Ex 8.03 (MyInteger)

Define a class named MyInteger that stores an integer and has functions to get and set the integer’s value. Then, overload the [] operator so that the index returns the digit in position i, where i = 0 is the least-significant digit. If no such digit exists then –1 should be returned. For example, if x is of type MyInteger and is set to 418, then x [0] should return 8, x [1] should return 1, x [2] should return 4, and x [3] should return –1.

Ex 8.04 (PrimeNumber)

Define a class named PrimeNumber that stores a prime number. The default constructor should set the prime number to 2. Add another constructor that allows the caller to set the prime number. Also, add a function to get the prime number. Finally, overload the prefix and postfix ++ and -- operators so they return a PrimeNumber object that is the next largest prime number (for ++) and the next smallest prime number (for --). For example, if the object's prime number is set to 13, then invoking ++ should return a PrimeNumber object whose prime number is set to 17. Create an appropriate test program for the class. Exercises for Introduction to C++ Programming Page 24 / 54

Chap 09 - Strings

Ex 9.01 (Correcting Grammar)

Write a program that will read in a sentence of up to 100 characters and output the sentence with spacing corrected and with letters corrected for capitalization. In other words, in the output sentence all strings of two or more blanks should be compressed to a single blank. The sentence should start with an uppercase letter but should contain no other uppercase letters. Do not worry about proper names; if their first letter is changed to lowercase, that is acceptable. Treat a line break as if it were a blank in the sense that a line break and any number of blanks are compressed to a single blank. Assume that the sentence ends with a period and contains no other periods. For example, the input the Answer to life, the Universe, and everything IS 42. should produce the following output:

The answer to life, the universe, and everything is 42.

Ex 9.02 (Letter Count)

Write a program that will read in a line of text and output the number of words in the line and the number of occurrences of each letter. Define a word to be any string of letters that is delimited at each end by either whitespace, a period, a comma, or the beginning or end of the line. You can assume that the input consists entirely of letters, whitespace, commas, and periods. When outputting the number of letters that occur in a line, be sure to count uppercase and lowercase versions of a letter as the same letter. Output the letters in alphabetical order and list only those letters that occur in the input line. For example, the input line

I say Hi. should produce output similar to the following:

3 words 1 a 1 h 2 i 1 s 1 y Exercises for Introduction to C++ Programming Page 25 / 54

Ex 9.03 (Pig Latin)

Write a program that converts a sentence input by the user into pig latin. You can assume that the sentence contains no punctuation. The rules for pig latin are as follows:

a. For words that begin with consonants, move the leading consonant to the end of the word and add “ay.” Thus, “ball” becomes “allbay”; “button” becomes “uttonbay”; and so forth.

b. For words that begin with vowels, add “yay” to the end. Thus, “all” becomes “allyay”; “one” becomes “oneyay”; and so forth.

Ex 9.04 (Anagrams)

Write a function that determines if two strings are anagrams. The function should not be case sensitive and should disregard any punctuation or spaces. Two strings are anagrams if the letters can be rearranged to form each other. For example, “Eleven plus two” is an anagram of “Twelve plus one”. Each string contains one “v”, three “e’s”, two “l’s”, etc. Test your function with several strings that are anagrams and non-anagrams. You may use either the string class or a C-style string. Exercises for Introduction to C++ Programming Page 26 / 54

Chap 10 - Pointers + Dynamic Arrays

Ex 10.01 (Polynomials)

Using dynamic arrays, implement a polynomial class with polynomial addition, subtraction, and multiplication.

Discussion: A variable in a polynomial does nothing but act as a placeholder for the coefficients. Hence, the only interesting thing about polynomials is the array of coefficients and the corresponding exponent. Think about the polynomial x*x*x + x + 1

Where is the term in x*x ? One simple way to implement the polynomial class is to use an array of doubles to store the coefficients. The index of the array is the exponent of the corresponding term. If a term is missing, then it simply has a zero coefficient.

There are techniques for representing polynomials of high degree with many missing terms. These use so-called sparse matrix techniques. Unless you already know these techniques, or learn very quickly, do not use these techniques.

Provide a default constructor, a copy constructor, and a parameterized constructor that enables an arbitrary polynomial to be constructed.

Supply an overloaded operator = and a destructor.

Provide these operations: polynomial + polynomial, constant + polynomial, polynomial + constant, polynomial - polynomial, constant - polynomial, polynomial - constant. polynomial * polynomial, constant * polynomial, polynomial * constant,

Supply functions to assign and extract coefficients, indexed by exponent.

Supply a function to evaluate the polynomial at a value of type double. You should decide whether to implement these functions as members, friends, or standalone functions.

Ex 10.02 (Reverse String) Exercises for Introduction to C++ Programming Page 27 / 54

Write a program that accepts a C-string input from the user and reverses the contents of the string. Your program should work by using two pointers. The “head” pointer should be set to the address of the first character in the string, and the “tail” pointer should be set to the address of the last character in the string (i.e., the character before the terminating null ). The program should swap the characters referenced by these pointers, increment “head” to point to the next character, decrement “tail” to point to the second-to-last character, and so on, until all characters have been swapped and the entire string reversed.

Ex 10.03 (Dynamic String Array)

One problem with dynamic arrays is that once the array is created using the new operator the size cannot be changed. For example, you might want to add or delete entries from the array similar to the behavior of a vector. This project asks you to create a class called DynamicStringArray that includes member functions that allow it to emulate the behavior of a vector of strings.

The class should include the following:

• A private member variable called dynamicArray that references a dynamic array of type string. • A private member variable called size that holds the number of entries in the array. • A default constructor that sets the dynamic array to NULL and sets size to 0. • A function that returns size. • A function named addEntry that takes a string as input. The function should create a new dynamic array one element larger than dynamicArray, copy all elements from dynamicArray into the new array, add the new string onto the end of the new array, increment size, delete the old dynamicArray, and then set dynamicArray to the new array. • A function named deleteEntry that takes a string as input. The function should search dynamicArray for the string. If not found, it returns false. If found, it creates a new dynamic array one element smaller than dynamicArray. It should copy all elements except the input string into the new array, delete dynamicArray, decrement size, and return true. • A function named getEntry that takes an integer as input and returns the string at that index in dynamicArray. It should return NULL if the index is out of dynamicArray’s bounds. • A copy constructor that makes a copy of the input object’s dynamic array. • Overload the assignment operator so that the dynamic array is properly copied to the target object. • A destructor that frees up the memory allocated to the dynamic array. Exercises for Introduction to C++ Programming Page 28 / 54

Embed your class in a suitable test program.

Chap 11 - Separate Compilation + Namespaces

Ex 11.01 (Separate Compilation)

This exercise is intended to illustrate namespaces and separate compilation in your development environment. You should use the development environment you regularly use in this course for this exercise. In a file f.h, place a declaration of void f( ) in namespace A. In a file g.h, place a declaration of void g( ) in namespace A. In files f.cpp and g.cpp, place the definitions of void f( ) and void g( ), respectively. Place the definitions of void f( ) and void g( ) in namespace A. The functions can do anything you want, but to keep track of execution include something like cout << "Function_Name called" << endl; where Function_Name is the name of the particular function. In another file, main.cpp, put your main function, #include the minimum collection of files to provide access to the names from namespace A. In your main function call the functions f then g. Compile, link, and execute using your development environment. To provide access to names in namespaces, you may use local using declarations such as using std::cout; or use local using directives such as using namespace std; inside a block, or qualify names using the names of namespaces, such as std::cout.

You may not use global namespace directives such as the following which are not in a block and apply to the entire file: using namespace std;

Of course you must handle namespace A and function names f and g, in addition to possibly std and cout. After doing this, write a one page description of how to create and use namespaces and separate compilation in your environment. Exercises for Introduction to C++ Programming Page 29 / 54

Ex 11.02 (Security)

You would like to verify the credentials of a user for your system. Listed next is a class named Security, which authenticates a user and password. (Note that this example is really not very secure. Typically passwords would be encrypted or stored in a database.) class Security { public: static int validate(string username, string password); };

// This subroutine hard-codes valid users and is not // considered a secure practice. // It returns 0 if the credentials are invalid, // 1 if valid user, and // 2 if valid administrator int Security::validate(string username, string password) { if ((username=="abbott") && (password=="monday")) return 1; if ((username=="costello") && (password=="tuesday")) return 2; return 0; }

Break this class into two files, a file with the header Security.h and a file with the implementation Security.cpp. Next, create two more classes that use the Security class by including the header file. The first class should be named Administrator and contain a function named Login that returns true if a given username and password have administrator clearance. The second class should be named User and contain a function named Login that returns true if a given username and password have either user or administrator clearance. Both the User and Administrator classes should be split into separate files for the header and implementation.

Finally, write a main function that invokes the Login function for both the User and Administrator classes to test if they work properly. The main function should be in a separate file. Be sure to use the #ifndef directive to ensure that no header file is included more than once.

Ex 11.03 (Unnamed Namespace)

This Programming Project explores how the unnamed namespace works. Listed are snippets from a program to perform input validation for a username Exercises for Introduction to C++ Programming Page 30 / 54 and password. The code to input and validate the username is in a separate file than the code to input and validate the password.

File header user.cpp: namespace Authenticate { void inputUserName( ) { do { cout << "Enter your username (8 letters only)" << endl; cin >> username; } while (!isValid( )); } string getUserName( ) { return username; } }

Define the username variable and the isValid( ) function in the unnamed namespace so the code will compile. The isValid( ) function should return true if username contains exactly eight letters. Generate an appropriate header file for this code.

Repeat the same steps for the file password.cpp, placing the password variable and the isValid( ) function in the unnamed namespace. In this case, the isValid( ) function should return true if the input password has at least eight characters including at least one non-letter:

File header password.cpp: namespace Authenticate { void inputPassword( ) { do { cout << "Enter your password (at least 8 characters " << "and at least one non-letter)" << endl; cin >> password; } while (!isValid( )); }

string getPassword( ) { return password; Exercises for Introduction to C++ Programming Page 31 / 54

} }

At this point, you should have two functions named isValid( ), each in different unnamed namespaces. Place the following main function in an appropriate place. The program should compile and run. int main( ) { inputUserName( ); inputPassword( ); cout << "Your username is " << getUserName( ) << " and your password is: " << getPassword( ) << endl; return 0; }

Test the program with several invalid usernames and passwords.

Exercises for Introduction to C++ Programming Page 32 / 54

Chap 12 - Streams + File I/O

Ex 12.01 (Largest Int)

Write a program that will search a file of numbers of type int and write the largest and the smallest numbers to the screen. The file contains nothing but numbers of type int separated by blanks or line breaks.

Ex 12.02 (File Merge)

Write a program that merges the numbers in two files and writes all the numbers into a third file. Your program takes input from two different files and writes its output to a third file. Each input file contains a list of numbers of type int in sorted order from the smallest to the largest. After the program is run, the output file will contain all the numbers in the two input files in one longer list in sorted order from smallest to largest. Your program should define a function that is called with the two input-file streams and the output-file stream as three arguments.

Ex 12.03 (Junk Mail)

Write a program to generate personalized junk mail. The program takes input both from an input file and from the keyboard. The input file contains the text of a letter, except that the name of the recipient is indicated by the three characters #N#. The program asks the user for a name and then writes the letter to a second file but with the three letters #N# replaced by the name. The three-letter string #N# will occur exactly once in the letter.

Hint: Have your program read from the input file until it encounters the three characters #N#, and have it copy what it reads to the output file as it goes. When it encounters the three letters #N#, it then sends output to the screen asking for the name from the keyboard. You should be able to figure out the rest of the details. Your program should define a function that is called with the input- and output file streams as arguments. If this is being done as a class assignment, obtain the file names from your instructor.

Harder version: Allow the string #N# to occur any number of times in the file. In this case the name is stored in two string variables. For this version assume that there is a first name and last name but no middle names or initials.

Ex 12.04 (Correcting a C++ Program) Exercises for Introduction to C++ Programming Page 33 / 54

Write a program that will correct a C++ program that has errors in which operator, << or >>, it uses with cin and cout. The program replaces each (i ncorrect) occurrence of

cin << with the corrected version

cin >> and each (incorrect) occurrence of

cout >> with the corrected version

cout <<

For an easier version, assume that there is always exactly one blank symbol between any occurrence of cin and a following <<, and similarly assume that there is always exactly one blank space between each occurrence of cout and a following >>.

For a harder version, allow for the possibility that there may be any number of blanks, even zero blanks, between cin and << and between cout and >>; in this harder case, the replacement corrected version has only one blank between the cin or cout and the following operator. The program to be corrected is in one file and the corrected version is output to a second file. Your program should define a function that is called with the input- and output-file streams as arguments.

Hint: Even if you are doing the harder version, you will probably find it easier and quicker to first do the easier version and then modify your program so that it performs the harder task.

Ex 12.05 (Answering Questions)

Write a program that allows the user to type in any one-line question and then answers that question. The program will not really pay any attention to the question, but will simply read the question line and discard all that it reads. It always gives one of the following answers:

I’m not sure but I think you will find the answer in Chapter #N. That's a good question. If I were you, I would not worry about such things. That question has puzzled philosophers for centuries. Exercises for Introduction to C++ Programming Page 34 / 54

I don't know. I'm just a machine. Think about it and the answer will come to you. I used to know the answer to that question, but I've forgotten it. The answer can be found in a secret place in the woods.

These answers are stored in a file (one answer per line), and your program simply reads the next answer from the file and writes it out as the answer to the question. After your program has read the entire file, it simply closes the file, reopens the file, and starts down the list of answers again.

Ex 12.06 (KWIX – Key Word In ConteXt)

In this program you are to process text to create a KWIX table (Key Word In conteXt table). The idea is to produce a list of keywords (not programming language keywords, rather words that have important technical meaning in a discussion), then for each instance of each keyword, place the keyword, the line number of the context, and the keyword in its context in the table. There may be more than one context for a given keyword. The sequence of entries within a keyword is to be the order of occurrence in the text. For this problem, “context” is a user-selected number of words before the keyword, the keyword itself, and a user-selected number of words after the keyword.

The table has an alphabetized column of keywords followed by a line number(s) where the keyword occurs, followed by a column of all contexts within which the keyword occurs. See the following example. For a choice of text consult your instructor.

Hints: To get your list of keywords, you should choose and type in several paragraphs from the text, then omit from your paragraph “boring” words such as forms of the verb “to be”; pronouns such as I, me, he, she, her, you, us, them, who, which, etc. Finally, sort the keyword list and remove duplicates. The better job you do at this, the more useful output you will get.

Example: A paragraph and its KWIX Listing: There are at least two complications when reading and writing with random access via an fstream: (1) You normally work in bytes using the type char or arrays of char and need to handle type conversions on your own, and (2) you typically need to position a pointer (indicating where the read or write begins) before each read or write.

KWIX Listing Keyword Line Number Keyword in Context access 2 with random access via arrays 3 char or arrays of bytes 2 work in bytes using char 3 the type char or char 3 array of char and conversions 3 handle type conversions on Exercises for Introduction to C++ Programming Page 35 / 54

Note: The table is longer than these sample entries.

Ex 12.07 (Highest Scores)

Write a class that tracks the five highest scores for a game. The scores should be stored in a file and include the player’s name as a string and the player’s score as an integer. The list of top scores should initially consist of the name Anonymous and scores of 0. The class should support the following functions:

■ A way to output to the screen the name and score of the top five players. The scores should be listed in order, with the highest score first and the lowest score last.

■ A function that takes a new name and score. If the score is higher than any of the top five scores then it should be added to the file and the lowest score discarded. Otherwise, the top list should remain unchanged.

Include an appropriate constructor or destructor if necessary, along with any helper functions. Write a main function that tests your class by simulating several score entries and outputting the high score list.

Ex 12.08 (Handling Bad Input)

One problem using cin to read directly into a variable such as an int is that if the user enters non-integer data then the program will continue with erroneous data and usually crash. A solution to this problem is to input data as a string, perform input validation, and then convert the string to an integer. Write a function that prompts the user to enter an integer. The program should use getline to read the user’s input into a string. Then use the stringstream class to extract an integer from the string. If an integer cannot be extracted then the user should be prompted to try again. The function should return the extracted integer. Exercises for Introduction to C++ Programming Page 36 / 54

Chap 13 - Recursion

Ex 13.01 (Number of Combinations)

The formula for computing the number of ways of choosing r different things from a set of n things is the following:

C ( n, r ) = n !/( r !*( n - r )!)

The factorial function n! is defined by

n ! = n *( n -1)*( n -2)*...*1

Discover a recursive version of the formula for C ( n, r ) and write a recursive function that computes the value of the formula. Embed the function in a program and test it.

Ex 13.02 (Towers of Hanoi)

Towers of Hanoi. There is a story about Buddhist monks who are playing this puzzle with 64 golden disks. The story claims that when the monks finish moving the disks from one post to a second via the third post, time will end. Eschatology (concerns about the end of time) and theology will be left to those better qualified; our interest is limited to the recursive solution to the problem.

A stack of n disks of decreasing size is placed on one of three posts. The task is to move the disks one at a time from the first post to the second. To do this, any disk can be moved from any post to any other post, subject to the rule that you can never place a larger disk over a smaller disk. The (spare) third post is provided to make the solution possible. Your task is to write a recursive function that describes instructions for a solution to this problem. We do not have graphics available, so you should output a sequence of instructions that will solve the problem.

Hint: If you could move n-1 of the disks from the first post to the third post using the second post as a spare, the last disk could be moved from the first post to the second post. Then by using the same technique (whatever that may be), you can move the n-1 disks from the third post to the second post, using the first post as a spare. There! You have the puzzle solved. You have only to decide what the nonrecursive case is, what the recursive case is, and when to output instructions to move the disks. Exercises for Introduction to C++ Programming Page 37 / 54

Ex 13.03 (Handshakes)

We have n people in a room, where n is an integer greater than or equal to 1. Each person shakes hands once with every other person. What is the total number, h(n), of handshakes? Write a recursive function to solve this problem. To get you started, if there are only one or two people in the room, then

handshake(1) = 0 handshake(2) = 1

If a third person enters the room, he or she must shake hands with each of the two people already there. This is two handshakes in addition to the number of handshakes that would be made in a room of two people, or a total of three handshakes.

If a fourth person enters the room, he or she must shake hands with each of the three people already there. This is three handshakes in addition to the number of handshakes that would be made in a room of three people, or six handshakes. If you can generalize this to n handshakes, you should be able to write the recursive solution. Exercises for Introduction to C++ Programming Page 38 / 54

Chap 14 - Inheritance

Ex 14.01 (Vehicle Class)

Create a base class called Vehicle that has the manufacturer’s name (type string ), number of cylinders in the engine (type int ), and owner (type Person given in the code that follows). Then create a class called Truck that is derived from Vehicle and has additional properties, the load capacity in tons (type double since it may contain a fractional part) and towing capacity in pounds (type int ). Be sure your classes have a reasonable complement of constructors and accessor methods, an overloaded assignment operator, and a copy constructor. Write a driver program that tests all your methods. The definition of the class Person follows. The implementation of the class is part of this programming project. class Person { public: Person( ); Person(string theName); Person(const Person& theObject); string getName( ) const; Person& operator=(const Person& rtSide); friend istream& operator >>(istream& inStream, Person& personObject); friend ostream& operator <<(ostream& outStream, const Person& personObject); private: string name; };

Ex 14.02 (Document Class)

Define a class named Document that contains a member variable of type string named text that stores any textual content for the document. Create a function named getText that returns the text field, a way to set this value, and an overloaded assignment operator.

Next, define a class for Email that is derived from Document and that includes member variables for the sender, recipient, and title of an e-mail message. Implement appropriate accessor and mutator functions. The body of the email message should be stored in the inherited variable text. Also overload the assignment operator for this class. Exercises for Introduction to C++ Programming Page 39 / 54

Similarly, define a class for File that is derived from Document and that includes a member variable for the pathname. Implement appropriate accessor and mutator functions for the pathname and overload the assignment operator.

Finally, create several sample objects of type Email and File in your main function. Test your objects by passing them to the following subroutine, which will return true if the object contains the specified keyword in the text property. bool ContainsKeyword( const Document& docObject, string keyword) { if (docObject.getText().find(keyword) != string::npos) return true; return false; }

For example, you might test to see whether an e-mail message contains the text "c++" with the call ContainsKeyword(emailObj, "c++");.

Ex 14.03 (Fantasy RPG)

Suppose that you are creating a fantasy role-playing game. In this game, we have four different types of creatures: humans, cyberdemons, balrogs, and elves. To represent one of these creatures, we might define a Creature class as follows: class Creature { private: int type; // 0 human, 1 cyberdemon, 2 balrog, 3 elf int strength; // How much damage we can inflict int hitpoints; // How much damage we can sustain string getSpecies(); // Returns type of species public: Creature( ); // Initialize to human, 10 strength, 10 hit points Creature(int newType, int newStrength, int newHit); // Initialize creature to new type, strength, hit points // Also add appropriate accessor and mutator functions // for type, strength, and hit points int getDamage(); // Returns amount of damage this creature // inflicts in one round of combat }; Exercises for Introduction to C++ Programming Page 40 / 54

Here is an implementation of the getSpecies( ) function: string Creature::getSpecies() { switch (type) { case 0: return "Human"; case 1: return "Cyberdemon"; case 2: return "Balrog"; case 3: return "Elf"; } return "Unknown"; }

The getDamage( ) function outputs and returns the damage this creature can inflict in one round of combat. The rules for calculating the damage are as follows:

■ Every creature inflicts damage that is a random number r, where 0 < r <= strength.

■ Demons have a 5% chance of inflicting a demonic attack, which is an additional 50 damage points. Balrogs and Cyberdemons are demons.

■ Elves have a 10% chance to inflict a magical attack that doubles the normal amount of damage.

■ Balrogs are very fast, so they get to attack twice.

An implementation of getDamage( ) is given here: int Creature::getDamage( ) { int damage; // All creatures inflict damage, which is a // random number up to their strength

damage = (rand( ) % strength) + 1; cout << getSpecies( ) << " attacks for " << damage << " points!" << endl; // Demons can inflict damage of 50 with a 5% chance if ((type = 2) || (type == 1)) if ((rand( ) % 100) < 5) { damage = damage + 50; cout << "Demonic attack inflicts 50 " << " additional damage points!" << endl; } Exercises for Introduction to C++ Programming Page 41 / 54

// Elves inflict double magical damage with a 10% chance if (type == 3) { if ((rand( ) % 10)==0) { cout << "Magical attack inflicts " << damage << " additional damage points!" << endl; damage = damage * 2; } } // Balrogs are so fast they get to attack twice if (type == 2) { int damage2 = (rand() % strength) + 1; cout << "Balrog speed attack inflicts " << damage2 << " additional damage points!" << endl; damage = damage + damage2; } return damage; }

One problem with this implementation is that it is unwieldy to add new creatures. Rewrite the class to use inheritance, which will eliminate the need for the variable type. The Creature class should be the base class. The classes Demon, Elf, and Human should be derived from Creature. The classes Cyberdemon and Balrog should be derived from Demon. You will need to rewrite the getSpecies( ) and getDamage( ) functions so they are appropriate for each class.

For example, the getDamage( ) function in each class should only compute the damage appropriate for that object. The total damage is then calculated by combining the results of getDamage( ) at each level of the inheritance hierarchy.

As an example, invoking getDamage( ) for a Balrog object should invoke getDamage( ) for the Demon object, which should invoke getDamage( ) for the Creature object. This will compute the basic damage that all creatures inflict, followed by the random 5% damage that demons inflict, followed by the double damage that balrogs inflict.

Also include mutator and accessor functions for the private variables. Write a main function that contains a driver to test your classes. It should create an object for each type of creature and repeatedly outputs the results of getDamage( ).

Ex 14.04 (Pets) Exercises for Introduction to C++ Programming Page 42 / 54

Define a Pet class that stores the pet’s name, age, and weight. Add appropriate constructors, accessor functions, and mutator functions. Also define a function named getLifespan that returns a string with the value “unknown lifespan.”

Next, define a Dog class that is derived from Pet. The Dog class should have a private member variable named breed that stores the breed of the dog. Add mutator and accessor functions for the breed variable and appropriate constructors. Redefine the getLifespan function to return “Approximately 7 years” if the dog’s weight is over 100 pounds and “Approximately 13 years” if the dog's weight is under 100 pounds.

Next, define a Rock class that is derived from Pet. Redefine the getLifespan function to return “Thousands of years”.

Finally, write a test program that creates instances of pet rocks and pet dogs that exercise the inherited and redefined functions. Exercises for Introduction to C++ Programming Page 43 / 54

Chap 15 - Polymorphism and Virtual Functions

Ex 15.01 (Figures) TODO – THE REST OF Chap 15

Consider a graphics system that has classes for various figures—say, rectangles, squares, triangles, circles, and so on. For example, a rectangle might have data members height, width, and center point, while a square and circle might have only a center point and an edge length or radius, respectively. In a well-designed system, these would be derived from a common class, Figure. You are to implement such a system.

The class Figure is the base class. You should add only Rectangle and Triangle classes derived from Figure. Each class has stubs for member functions erase and draw. Each of these member functions outputs a message telling what function has been called and what the class of the calling object is. Since these are just stubs, they do nothing more than output this message. The member function center calls erase and draw to erase and redraw the figure at the center. Because you have only stubs for erase and draw, center will not do any “centering” but will call the member functions erase and draw. Also, add an output message in the member function center that announces that center is being called. The member functions should take no arguments. There are three parts to this project:

a. Do the class definitions using no virtual functions. Compile and test. b. Make the base class member functions virtual. Compile and test. c. Explain the difference in results.

For a real example, you would have to replace the definition of each of these member functions with code to do the actual drawing. Use the following main function for all testing:

#include #include "figure.h" #include "rectangle.h" #include "triangle.h" using namespace std; int main( ) { Triangle tri; tri.draw( ); cout << "\nDerived class Triangle object calling center( ).\n"; tri.center( ); //Calls draw and center Rectangle rect; rect.draw( ); Exercises for Introduction to C++ Programming Page 44 / 54

cout << "\nDerived class Rectangle object calling center( ).\n"; rect.center( ); //Calls draw and center return 0; } Exercises for Introduction to C++ Programming Page 45 / 54

Chap 16 - Templates

Ex 16.01 (Template binary search)

Write a template version of the iterative binary search algorithm which only searches an array of integers for an integer key. Specify requirements on the template parameter type. Extend your function so that it accepts any type.

Ex 16.02 (Template insertion sort)

The template sort routine in Display 16.3 is based on an algorithm called the selection sort. Another related sorting algorithm is called insertion sort. The insertion sort algorithm is the sort method often used to sort a Bridge hand. Consider each element in turn, inserting it into its proper place among the elements at the start of the array that are already sorted. The element being considered is inserted by moving the larger elements “to the right” to make space and inserting the vacated place. For example, the following shows the steps in a selection sort of an array of int’s a. The values of a[0] through a[4] are given on each line. The asterisk marks the boundary between the sorted and unsorted portions of the array.

2 * 5 3 4 2 5 * 3 4 2 3 5 * 4 2 3 4 5 *

First, write an insertion sort function that works for int’s. Next, write the template version of this sort function. Finally, test thoroughly using several primitive types, using a type you create with the minimal machinery necessary to use the sort routine.

Ex 16.03 (Template sets)

Write a template-based class that implements a set of items. The class should allow the user to

a. Add a new item to the set. b. Get the number of items in the set. c. Get a pointer to a dynamically created array containing each item in the set. Exercises for Introduction to C++ Programming Page 46 / 54

The caller of this function is responsible for deallocating the memory. Test your class by creating sets of different data types (e.g., integers, strings).

Chap 17 - Templates

Ex 17.01 (Reverse linked list)

Write a void function that takes a linked list of integers and reverses the order of its nodes. The function will have one call-by-reference parameter that is a pointer to the head of the list. After the function is called, this pointer will point to the head of a linked list that has the same nodes as the original list but in the reverse of the order they had in the original list. Note that your function will neither create nor destroy any nodes. It will simply rearrange nodes. Place your function in a suitable test program.

Ex 17.02 (Merge linked lists)

Write a function called mergeLists that takes two call-by-reference arguments that are pointer variables that point to the heads of linked lists of values of type int. The two linked lists are assumed to be sorted so that the number at the head is the smallest number, the number in the next node is the next smallest, and so forth. The function returns a pointer to the head of a new linked list that contains all the nodes in the original two lists. The nodes in this longer list are also sorted from smallest to largest values. Note that your function will neither create nor destroy any nodes. When the function call ends, the two pointer variable arguments should have the value NULL.

Ex 17.04 (The beautiful princess (again))

In an ancient land, the beautiful princess Eve had many suitors. She decided on the following procedure to determine which suitor she would marry. First, all of the suitors would be lined up one after the other and assigned numbers. The first suitor would be number 1, the second number 2, and so on up to the last suitor, number n. Starting at the first suitor, she would then count three suitors down the line (because of the three letters in her name) and the third suitor would be eliminated from winning her hand and removed from the line. Eve would then continue, counting three more suitors, and eliminating every third suitor. When she reached the end of the line, she would continue counting from the beginning. For example, if there were six suitors, then the elimination process would proceed as follows:

123456 Initial list of suitors, start counting from 1 Exercises for Introduction to C++ Programming Page 47 / 54

12456 Suitor 3 eliminated, continue counting from 4 1245 Suitor 6 eliminated, continue counting from 1 125 Suitor 4 eliminated, continue counting from 5 15 Suitor 2 eliminated, continue counting from 5 1 Suitor 5 eliminated, 1 is the lucky winner

Write a program that creates a circular linked list of nodes to determine which position you should stand in to marry the princess if there are n suitors. Your program should simulate the elimination process by deleting the node that corresponds to the suitor that is eliminated for each step in the process. Be careful that you do not leave any memory leaks

Ex 17.05 (Graphs)

The following figure is called a graph. The circles are called nodes and the lines are called edges. An edge connects two nodes. You can interpret the graph as a maze of rooms and passages. The nodes can be thought of as rooms and an edge connects one room to another. Note that each node has at most four edges in the following graph.

Write a program that implements the previous maze using references to instances of a Node class. Each node in the graph will correspond to an instance of Node. The edges correspond to links that connect one node to another and can be represented in Node as instance variables that reference another Node class. Start the user in node A. The user’s goal is to reach the finish in node L. The program should output possible moves in the north, south, east, or west direction. Sample execution is shown next.

You are in room A of a maze of twisty little passages, all alike. You can go east or south. E You are in room B of a maze of twisty little passages, all alike. You can go west or south. S Exercises for Introduction to C++ Programming Page 48 / 54

You are in room F of a maze of twisty little passages, all alike. You can go north or east. E

Ex 17.06 (Graph search)

First, complete Ex 17.06. Then, write a recursive algorithm that finds a path from node A to node L. Your algorithm should work with any pair of start and finish nodes, not just nodes A and L. Your algorithm should also work if there are loops such as a connection between nodes E and F. Exercises for Introduction to C++ Programming Page 49 / 54

Chap 18 - Exception handling

Ex 18.01 (Function calls)

(Based on a problem in Stroustrup, The C++ Programming Language, 3rd edition) Write a program consisting of functions calling one another to a calling depth of 10. Give each function an argument that specifies the level at which it is to throw an exception. The main function prompts for and receives input that specifies the calling depth (level) at which an exception will be thrown. The main function then calls the first function. The main function catches the exception and displays the level at which the exception was thrown. Do not forget the case in which the depth is 0, where main must both throw and catch the exception.

Hints: You could use ten different functions or ten copies of the same function that call one another, but do not do this. Rather, for compact code, use a main function that calls another function that calls itself recursively. Suppose you do this; is the restriction on the calling depth necessary? This can be done without giving the function any additional arguments, but if you cannot do it that way, try adding an additional argument to the function.

Ex 18.02 (Function calls)

A function that returns a special error code is usually better accomplished throwing an exception instead. The following class maintains an account balance. class Account { private: double balance; public: Account() { balance = 0; }

Account( double initialDeposit) { balance = initialDeposit; }

double getBalance() { Exercises for Introduction to C++ Programming Page 50 / 54

return balance; }

// returns new balance or -1 if error double deposit( double amount) { if (amount > 0) balance += amount; else return -1; // Code indicating error return balance; }

// returns new balance or -1 if invalid amount double withdraw( double amount) { if ((amount > balance) || (amount < 0)) return -1; else balance -= amount; return balance; } };

Rewrite the class so that it throws appropriate exceptions instead of returning -1 as an error code. Write test code that attempts to withdraw and deposit invalid amounts and catches the exceptions that are thrown. Exercises for Introduction to C++ Programming Page 51 / 54

Chap 19 - Standard Template Library

Ex 19.01 (Iterators)

The point of this exercise is to demonstrate that an object that behaves like an iterator is an iterator. More precisely, if an object accesses some container and behaves like an iterator of a particular strength, then that object can be used as an iterator to manipulate the container with any of the generic functions that require an iterator having that strength. However, while the generic algorithms can be used with this container, the member functions, such as begin and end, will (of course) not be present (for example, in an array) unless they have been explicitly coded for that container. We will restrict this exercise to arrays of double, but the same message would hold true for any base type. a. Argue from the properties of random-access iterators that a pointer that points to an element of an array behaves exactly as a random-access iterator. b. Argue further that

i) the name of the array is a pointer to double that points to the first element and so the array name can serve as a “begin” iterator and

ii) (the array’s name) + (the size of the array) can serve as an “end” pointer. (Of course, this points one past the end, as it should.) c. Write a short program in which you declare an array of double of size 10, and populate this array with 10 double’s. Then call the sort generic algorithm with these pointer values as arguments and display the results.

Ex 19.02 (Removal)

This problem intends to illustrate removal of several instances of a particular item from a container with the remove generic function. A side effect is to examine the behavior of the generic function remove that comes with your compiler. (We have observed some variation in the behavior of remove from one compiler to another.) Before you start, look up the behavior of the remove generic algorithm as described by your favorite STL document or Web site. (For example, point your browser at http://www.sgi.com/tech/stl/remove.html. This worked as of the publication date.) a. Modify the array declaration in Programming Project 19.1 to include several elements with the same value (say, 4.0, but you can use any value for this Exercises for Introduction to C++ Programming Page 52 / 54 exercise). Make sure that some of these are not all together. Use the modifying generic function remove (see Display 19.19 ) to remove all elements 4.0 (that is, the value you duplicated in building the array). b. Use the array of double from part a to build a list and a vector that have the same contents as the array. Do this using the container constructor that takes two iterators as arguments. The vector and list classes each have a constructor that takes two iterators as arguments and initializes the vector or list to the items in the iterator interval. The iterators can be located in any container, including in an array. Build the vector and list using the array name for the begin iterator and the name + array length as the end iterator for the two constructor arguments. Use the modifying algorithm remove to remove from the list and from the vector all elements equal to 4.0 (or the value you duplicated in building the array). Display the contents of the vector and list and explain the results. c. Modify the code from part b to assign to an iterator variable of appropriate type the iterator value returned by the call to the remove generic function. Review the documentation for remove to be certain you know what these iterator values mean. Output the contents of the array, the vector and the list, begin( ) to end( ), using the “begin” and “end” we described previously for the array. Output the contents of the two containers starting at the iterator returned from remove to the end() of each container. Explain your results.

Ex 19.03 (Sieve of Eratosthenes)

A prime number is an integer greater than 1 and divisible only by itself and 1. An integer x is divisible by an integer y if there is another integer z such x = y*z. The Greek mathematician Eratosthenes (pronounced Er-ah - tos-thin- eeze) gave an algorithm for finding all prime numbers less than some integer N. This algorithm is called the Sieve of Eratosthenes. It works like this: Begin with a list of integers 2 through N. The number 2 is the first prime. (It is instructive to consider why this is true.) The multiples of 2—that is, 4, 6, 8, etc.—are not prime. We cross these off the list. Then the first number after 2 that was not crossed off, which is 3, is the next prime. The multiples of 3 are not primes. Cross these off the list. Note that 6 is already gone, cross off 9, 12 is already gone, cross off 15, etc. The first number not crossed off is the next prime. The algorithm continues on this fashion until we reach N. All the numbers not crossed off the list are primes. a. Write a program using this algorithm to find all primes less than a user- supplied number N. Use a vector container for the integers. Use an array of bool initially set to all true to keep track of crossed off integers. Change the entry to false for integers that are crossed off the list. b. Test for N = 10, 30, 100, and 300. Exercises for Introduction to C++ Programming Page 53 / 54

Improvements: c. Actually, we do not need to go all the way to N. You can stop at N√2. Try this and test your program. N√2 works and is better, but is not the smallest number we could use. Argue that to get all the primes between 1 and N the minimum limit is the square root of N. d. Modify your code from part a to use the square root of N as an upper limit.

Ex 19.04 (Prime)

Here is pseudocode for a program that inputs a value n from the user and then inserts n random numbers, ensuring that there are no duplicates:

Input n from user Create vector v of type int Loop i = 1 to n r = random integer between 0 and n-1 Linearly search through v for value r if r is not in vector v then add r to the end of v End Loop Print number of elements added to v a. Implement this program with your own linear search routine and add wrapper code that will time how long it takes to run. Test the program for different values of n . Depending on the speed of your system, you may need to input large values for n so that the program takes at least one second to run. Here is a fragment that shows how to calculate the difference in time: (time.h is a library that should be available on your version of C ++ ).

#include

time_t start, end; double dif;

time(&start); // Record start time // Rest of program goes here. time(&end); // Record end time dif = difftime(end, start); cout << "It took " << dif << " seconds to execute. " << endl; b. Next, create a second program that has the same behavior except that it uses an STL set to store the numbers instead of a vector:

Input n from user Create set s of type int Loop i = 1 to n Exercises for Introduction to C++ Programming Page 54 / 54

r = random integer between 0 to n-1 Use s.find(r) to search if r is already in the set if r is not in set s then add r to s End Loop Print number of elements added to s

Time your new program with the same values of n that you used in the vector version. What do the results tell you about the Big- O run time of the find( ) function for the set compared with linear search through the vector? Note that the find( ) function is really redundant because insert has no effect if the element is already in the set. However, use the find( ) function anyway to create a program comparable to the vector algorithm.

Ex 19.05 (sxb)

Recommended publications