Professional Documents
Culture Documents
An Overview of C++
Out of these various experiments come programs. This is
our experience: programs do not come out of the minds of
one person or two people such as ourselves, but out of dayto-day work.
Stokely Carmichael and Charles V. Hamilton,
Black Power, 1967
Objectives
To recognize the components of a typical C++ program.
To understand the predefined data types in C++ and how they can be used to store
information in a program.
To become familiar with the facilities in the simplified and standard input/output (I/O)
libraries for reading input data and displaying results.
To understand the structure of expressions in C++ and how to use the common
operators to express calculations.
To recognize the statement forms if, switch, while, and for and be able to use them
in simple programs.
To be able to write simple programs that integrate the various control facilities
presented in this chapter.
An Overview of C++
In Lewis Carrolls Alices Adventures in Wonderland, the King asks the White Rabbit to
begin at the beginning and go on till you come to the end: then stop. Good advice, but
only if youre starting from the beginning. This book is designed for a second course in
computer science and therefore assumes that you have already begun your study of
programming. At the same time, because first courses vary considerably in what they
cover, it is difficult to rely on any specific material. Some of you, for example, will
already have experience programming in C or C++. Many of you, however, are coming
from a first course taught in some other language.
Because of this wide disparity in background, the best approach is to adopt the Kings
advice and begin at the beginning. The first three chapters in this text therefore move
quickly through the material I consider to be essential background for the later chapters.
Chapters 1 and 2 discuss C++ in general and may be skimmed if youve had experience
with C++. Chapter 3 discusses standard interfaces and some interfaces particular to this
text. By the end of these three chapters, you will be up to speed on the fundamentals of
C++ programming.
An Overview of C++
As Figure 1-1 illustrates, C++ represents the coming together of two branches in the
evolution of programming languages. One of its ancestors is a language called C, which
was designed at Bell Laboratories by Dennis Ritchie in 1972 and then later revised and
standardized by the American National Standards Institute (ANSI) in 1989. But C++ also
descends from another line of languages that have dramatically changed the nature of
modern programming.
The object-oriented paradigm
Over the last decade or so, computer science and programming have gone through
something of a revolution. Like most revolutionswhether political upheavals or the
conceptual restructurings that Thomas Kuhn describes in his 1962 book The Structure of
Scientific Revolutionsthis change has been driven by the emergence of an idea that
challenges an existing orthodoxy. Initially, the two ideas compete. For a while, the old
order maintains its dominance. Over time, however, the strength and popularity of the
new idea grows, until it begins to displace the older idea in what Kuhn calls a paradigm
shift. In programming, the old order is represented by the procedural paradigm, in
which programs consist of a collection of procedures and functions that operate on data.
The challenger is the object-oriented paradigm, in which programs are viewed instead
as a collection of data objects that exhibit particular behavior.
The idea of object-oriented programming is not really all that new. The first objectoriented language was SIMULA, a language for coding simulations designed in 1967 by
the Scandinavian computer scientists Ole-Johan Dahl, Bjrn Myhrhaug, and Kristen
Nygaard. With a design that was far ahead of its time, SIMULA anticipated many of the
concepts that later became commonplace in programming, including the concept of
abstract data types and much of the modern object-oriented paradigm. In fact, most of
the terminology used to describe object-oriented systems comes from the original 1967
report on SIMULA.
For many years, however, SIMULA mostly just sat on the shelf. Few people paid
much attention to it, and the only place you were likely to hear about it would be in a
course on programming language design. The first object-oriented language to gain any
significant level of recognition within the computing profession was Smalltalk, which
was developed at the Xerox Palo Alto Research Center (more commonly known as Xerox
PARC) in the late 1970s. The purpose of Smalltalk, which is described in the book
Smalltalk-80: The Language and Its Implementation by Adele Goldberg and David
Robson, was to make programming accessible to a wider audience. As such, Smalltalk
was part of a larger effort at Xerox PARC that gave rise to much of the modern userinterface technology that is now standard on personal-computers.
Despite many attractive features and a highly interactive user environment that
simplifies the programming process, Smalltalk never achieved much commercial success.
The profession as a whole took an interest in object-oriented programming only when the
central ideas were incorporated into variants of C, which had already become an industry
standard. Although there were several parallel efforts to design an object-oriented
language based on C, the most successful was the language C++, which was designed in
the early 1980s by Bjarne Stroustrup at AT&T Bell Laboratories. By making it possible
to integrate object-oriented techniques with existing C code, C++ enabled large
communities of programmers to adopt the object-oriented paradigm in a gradual,
evolutionary way.
Although object-oriented languages are undeniably gaining popularity at the expense
of procedural ones, it would be a mistake to regard the object-oriented and procedural
paradigms as mutually exclusive. Programming paradigms are not so much competitive
An Overview of C++
object file
/* File: count.c */
0100100101011001000
1000010100011101011
0110100111010101100
#include <stdio.h>
#include "genlib.h"
#define N 10
main()
{
int i;
compiler
executable file
0100100101011001000
1000010100011101011
0110100111010101100
1001011010110001011
0100100101001011011
0101101011010100101
other object
files/libraries
1001011010110001011
0100100101001011011
0101101011010100101
linker
An Overview of C++
As the annotations in Figure 1-3 indicate, the powertab.cpp program is divided into
several components, which are discussed in the next few sections.
Comments
Much of the text in Figure 1-3 consists of English-language comments. A comment is
text that is ignored by the compiler but which nonetheless conveys information to other
programmers. A comment consists of text enclosed between the markers /* and */ and
may continue over several lines. Alternatively, a single-line comment is begun by the
marker // and continues until the end of the line. The powertab.cpp program includes a
comment at the beginning that describes the operation of the program as a whole, one
before the definition of the RaiseIntToPower function that describes what it does, and a
couple of one-line comments that act very much like section headings in English text.
Library inclusions
The lines beginning with #include such as
#include "genlib.h"
#include <iostream>
#include <iomanip>
indicate that the compiler should read in definitions from a header file. The inclusion of
a header file indicates that the program uses facilities from a library, which is a
collection of prewritten tools that perform a set of useful operations. The different
punctuation in these #include lines reflects the fact that the libraries come from different
sources. The angle brackets are used to specify a system library, such as the standard
input/output stream library (iostream) or the stream manipuator library (iomanip) that is
supplied along with C++. The quotation marks are used for private libraries, including
the general library (genlib), which was designed for use with the programs in this text.
Every program in this book will include at least this library most will require other
libraries as well and must contain an #include line for each one.
An Overview of C++
program
comment
#include "genlib.h"
#include <iostream>
#include <iomanip>
library
inclusions
/*
* Constants
* --------* LOWER_LIMIT -- Starting value for the table
* UPPER_LIMIT -- Final value for the table
*/
section
comment
constant
definitions
function
prototype
main
program
function
comment
local variable
declarations
statements
forming body
of function
function
definition
An Overview of C++
Program-level definitions
After the #include lines for the libraries, many programs define constants that apply to
the program as a whole. In the powertab.cpp program, the following lines
const int LOWER_LIMIT = 0;
const int UPPER_LIMIT = 12;
which defines the constant name to be of type type and initialized to value. A constant must
be initialized when it is defined and once initialized, it cannot be assigned a new value or
changed in any way. Attempting to do so will result in a compiler error. After a named
constant is defined, it is available to be used anywhere in the rest of the program. For
example, after encountering the line
const double PI = 3.14159265;
any subsequent use of the name PI refers to the constant value 3.14159265.
Giving symbolic names to constants has several important advantages in terms of
programming style. First, the descriptive names give readers of the program a better
sense of what the constant value means. Second, centralizing such definitions at the top
of the file makes it easier to change the value associated with a name. For example, all
you need to do to change the limits used for the table in the powertab.cpp program is
change the values of the constants. And lastly, a const declaration protects from the
value from any unintended modification.
In addition to constants, programs often define new data types in this section of the
source file, as you will see in Chapter 2.
Function prototypes
Computation in a C++ program is carried out in the context of functions. A function is a
unit of code that (1) performs a specific operation and (2) is identified by name. The
powertab.cpp program contains two functionsmain and RaiseIntToPowerwhich
are described in more detail in the next two sections. The line
int RaiseIntToPower(int n, int k);
An Overview of C++
This code is an example a for loop, which is used to specify repetition. In this case, the
for statement indicates that the body of the loop should be repeated for each of the
values of n from LOWER_LIMIT to UPPER_LIMIT. A section on the detailed structure of
the for loop appears later in the chapter, but the example shown here represents a
common idiomatic pattern that you can use to count between any specified limits.
The body of the loop illustrates an important new feature: the ability to include values
as part of the output display. Rather than just displaying fixed strings, we can display
numeric values and computed results. We can also use stream manipulators to format the
output. Lets examine the first statement in the body of the loop:
cout << setw(3) << n << " |" ;
This line will display the current value of the variable n followed by a string
containing a space and a vertical bar. The setw(3) that is inserted into the stream just
before n indicates that the stream should format the next value in a field that is three
characters wide. Similarly, the next statement prints the formatted result taken from the
expression RaiseIntToPower(n, 2) . Obtaining the value of the expression requires
making a call on the RaiseIntToPower function, which is discussed in the following
section. The value that RaiseIntToPower returns is displayed as part of the output. So
is the result of the call to RaiseIntToPower(2, n), which supplies the value for the third
column in the table.
The last statement in main is
return 0;
which indicates that the function result is 0. The return value from the main function is
used to communicate the success or failure of the entire program. By convention, a result
of 0 indicates success.
An Overview of C++
Function definitions
Because large programs are difficult to understand in their entirety, most programs are
broken down into several smaller functions, each of which is easier to understand. In the
powertab.cpp program, the function RaiseIntToPower is responsible for raising an
integer to a poweran operation that is not built into C++ and must therefore be defined
explicitly.
The first line of RaiseIntToPower is the variable declaration
int result;
which introduces a new variable named result capable of holding values of type int,
the standard type used to represent integers. The syntax of variable declarations is
discussed in more detail in the section on Variables, values, and types later in this
chapter. For now, all you need to know is that this declaration creates space for an
integer variable that you can then use in the body of the function.
The next line in the body of RaiseIntToPower is
result = 1;
This statement is a simple example of an assignment statement, which sets the variable
on the left of the equal sign to the value of the expression on the right. In this case, the
statement sets the variable result to the constant 1. The next statement in the function is
a for loop that executes its body k times. The repeated code consists of the line
result *= n;
which is a C++ shorthand for the English sentence Multiply result by n. Because the
function initializes the value of result to 1 and then multiplies result by n a total of k
times, the variable result ends up with the value nk.
The last statement in RaiseIntToPower is
return result;
which indicates that the function should return result as the value of the function.
An Overview of C++
10
where type indicates the data type and namelist is a list of variable names separated by
commas. For example, the function RaiseIntToPower in the powertab.cpp program
contains the line
int result;
is a shorthand for the following code, in which the declaration and assignment are
separate:
int result;
result = 0;
An Overview of C++
Table 1-1 C++ reserved keywords
asm
do
auto
double
bool
dynamic_cast
break
else
case
enum
catch
explicit
char
extern
class
false
const
float
const_cast for
continue
friend
default
goto
delete
if
11
inline
int
long
mutable
namespace
new
operator
private
protected
public
register
reinterpret_cast
return
short
signed
sizeof
static
static_cast
struct
switch
template
this
throw
true
try
typedef
typeid
typename
union
unsigned
using
virtual
void
volatile
wchar_t
while
with a lowercase letter, such as n1, total, or string. By contrast, function names, such
as RaiseIntToPower, usually begin with an uppercase letter. Moreover, whenever a
name consists of several English words run together, the first letter in each word after the
first is capitalized to make the name easier to read. By tradition, constant names, such as
LOWER_LIMIT are written entirely in uppercase, with underscores between the words.
Local and global variables
Most variables are declared with the body of a function. Such variables are called local
variables. The scope of a local variable extends to the end of the block in which it is
declared. The lifetime of a local variable is the time during which that function is active.
When the function is called, space for each local variable is allocated for the duration of
that function call. When the function returns, all its local variables disappear.
If a variable declaration appears outside any function definition, that declaration
introduces a global variable. The scope of a global variable is the remainder of the file
in which it is declared. Its lifetime continues throughout the entire execution of a
program. Global variables are therefore able to store values that persist across function
calls. Although they have important applications, global variables can easily be
overused. Because global variables can be manipulated by many different functions, it is
harder to keep those functions from interfering with each other. Because of these
dangers, global variables are used infrequently in this text.
The concept of a data type
One of the reasons C++ requires all variables to be declared is that doing so constrains
their contents to values of a particular data type. From a formal perspective, a data type
is defined by two properties: a domain, which is the set of values that belong to that type,
and a set of operations, which defines the behavior of that type. For example, the domain
of the type int includes all integers (. . . 2, 1, 0, 1, 2 . . .) up to the limits established by
the hardware of the machine. The set of operations applicable to values of type int
includes, for example, the standard arithmetic operations like addition and multiplication.
Other types have a different domain and set of operations.
As you will learn in Chapter 2, much of the power of higher-level languages like C++
comes from the fact that you can define new data types from existing ones. To get that
process started, C++ includes several fundamental types that are defined as part of the
language (or, in the case of string, as part of the standard C++ libraries). These types,
which act as the building blocks for the type system as a whole, are called atomic types.
These predefined types are grouped into four categoriesinteger, floating-point, text, and
Booleanwhich are discussed in the sections that follow.
An Overview of C++
12
Integer types
Although the concept of an integer seems like a simple one, C++ actually includes several
different data types for representing integer values. In most cases, all you need to know
is the type int, which corresponds to the standard representation of an integer on the
computer system you are using. In certain cases, however, you need to be more careful.
Like all data, values of type int are stored internally in storage units that have a limited
capacity. Those values therefore have a maximum size, which limits the range of
integers you can use. To get around this problem, C++ defines three integer types
short, int, and longdistinguished from each other by the size of their domains.
Unfortunately, the language definition for C++ does not specify an exact range for
these three types. As a result, the range for the different integer types depends on the
machine and the compiler youre using. On many personal computers, the maximum
value of type int is 32,767, which is rather small by computational standards. If you
wanted, for example, to perform a calculation involving the number of seconds in a year,
you could not use type int on those machines, because that value (31,536,000) is
considerably larger than the largest available value of type int. The only properties you
can rely on are the following:
The internal size of an integer cannot decrease as you move from short to int to
long. A compiler designer for C++ could, for example, decide to make short and int
the same size but could not make int smaller than short.
The maximum value of type int must be at least 32,767 (2151).
The maximum value of type long must be at least 2,147,483,647 (2311).
The designers of C++ could have chosen to define the allowable range of type int
more precisely. For example, they could have declaredas the designers of Java did
that the maximum value of type int would be 2311 on every machine. Had they done
so, it would be easier to move a program from one system to another and have it behave
in the same way. The ability to move a program between different machines is called
portability, which is an important consideration in the design of a programming
language.
In C++, each of the integer types int , long , and short may be preceded by the
keyword unsigned . Adding u n s i g n e d creates a new data type in which only
nonnegative values are allowed. Because unsigned variables do not need to represent
negative values, declaring a variable to be one of the unsigned types allows it to hold
twice as many positive values. For example, if the maximum value of type int is 32,767,
the maximum value of type unsigned int will be 65,535. C++ allows the type unsigned
int to be abbreviated to unsigned , and most programmers who use this type tend to
follow this practice. Sometimes variables intended to store sizes are declared as
unsigned, because a size will always be nonnegative.
An integer constant is ordinarily written as a string of digits representing a number in
base 10. If the number begins with the digit 0, however, the compiler interprets the value
as an octal (base 8) integer. Thus, the constant 040 is taken to be in octal and represents
the decimal number 32. If you prefix a numeric constant with the characters 0x, the
compiler interprets that number as hexadecimal (base 16). Thus, the constant 0xFF is
equivalent to the decimal constant 255. You can explicitly indicate that an integer
constant is of type long by adding the letter L at the end of the digit string. Thus, the
constant 0L is equal to 0, but the value is explicitly of type long. Similarly, if you use the
letter U as a suffix, the constant is taken to be unsigned.
An Overview of C++
13
Floating-point types
Numbers that include a decimal fraction are called floating-point numbers, which are
used to approximate real numbers in mathematics. As with integers, C++ defines three
different floating-point types: float, double, and long double. Although ANSI C++
does not specify the exact representation of these types, the way to think about the
difference is that types that appear later in the list allow numbers to be represented with
greater precision but require more memory space. Unless you are doing exacting
scientific calculation, the differences between these types will not matter a great deal. In
keeping with a common convention among C++ programmers, this text uses the type
double as its standard floating-point type.
Floating-point constants in C++ are written with a decimal point. Thus, if 2.0 appears
in a program, the number is represented internally as a floating-point value if the
programmer had written 2, this value would be an integer. Floating-point values can also
be written in a special programmers style of scientific notation, in which the value is
represented as a floating-point number multiplied by a integral power of 10. To write a
number using this style, you write a floating-point number in standard notation, followed
immediately by the letter E and an integer exponent, optionally preceded by a + or - sign.
For example, the speed of light in meters per second can be written in C++ as
2.9979E+8
An Overview of C++
Table 1-2 ASCII codes
0
1
0
\000 \001
14
2
\002
3
\003
4
\004
5
\005
6
\006
7
\a
8
\b
9
\t
10
\n
\v
\f
\r
\016
\017
\020
\021
\022
\023
20
\024
\025
\026
\027
\030
\031
\032
\033
\034
\035
30
\036
\037
space
"
&
'
40
50
60
<
>
70
80
90
100
110
120
\177
escape sequences and you can see several listed in the first few rows of Table 1-2. A
few of the more commonly used escape sequences are '\n' the newline character,'\t'
the tab character, and '\\' the backslash character.
Characters are most useful when they are collected together into sequential units. In
programming, a sequence of characters is called a string. Strings make it possible to
display informational messages on the screen. You have already seen strings in the
sample program powertab.cpp. It is important, however, to recognize that strings are
data and that they can be manipulated and stored in much the same way that numbers can.
The standard C++ library defines a string type and operations that manipulate strings.
The details of type string are not important at this point strings are considered in more
detail in Chapter 3. In this chapter, strings are treated as atomic values and used
exclusively to specify text that is displayed directly on the display screen.
You write string constants in C++ by enclosing the characters contained within the
string in double quotes. C++ supports the same escape sequences for strings as for
characters. If two or more string constants appear consecutively in a program, the
compiler concatenates them together. The most important implication of this rule is that
you can break a long string over several lines so that it doesnt end up running past the
right margin of your program.
Boolean type
In the programs you write, it is often necessary to test a particular condition that affects
the subsequent behavior of your code. Typically, that condition is specified using an
expression whose value is either true or false. This data typefor which the only legal
values are true and falseis called Boolean data, after the mathematician George Boole,
who developed an algebraic approach for working with such values.
In C++, the Boolean type is called bool and its domain consists of the values true and
You can declare variables of type bool and manipulate them in the same way as
other data objects.
false.
An Overview of C++
15
The simpio library defines the functions GetInteger , GetLong , GetReal , and
GetLine, which wait for the user to enter a line at the keyboard and then return a value of
type int, long, double, and string, respectively. To let the user know what value is
expected, it is conventional to display a message to the user, which is called a prompt,
before calling the input function. Thus, if you need to request a value from the user for
the integer variable n, you would typically use a pair of statements like this:
cout << "Enter an integer: ";
n = GetInteger();
Output operations in this book use the insertion operator <<. The operand on the left of
the operator is a stream, such as the standard output stream cout. The operand on the
right is the data that you wish to insert into the stream. Several insertions to the same
stream can be chained together as shown here:
cout << "The result is " << val << endl;
Stream manipulators can be used to control the formatting of the output. A manipulator
is inserted into the stream ahead of the value it affects. The manipulator does not print
anything to the stream, but changes the state of the stream such that subsequent insertions
will use the requested formatting. A few of the more common stream manipulators are
shown in Table 1-3. To use these manipulators, you must include the <iomanip>
interface file in the library-inclusion section of your program.
As an example of the use of the simple I/O facilities, the following main program reads
in three floating-point values and displays their average:
int main() {
cout << "This program averages three numbers." << endl;
cout << "1st number: ";
double n1 = GetReal();
cout << "2nd number: ";
double n2 = GetReal();
cout << "3rd number: ";
double n3 = GetReal();
double average = (n1 + n2 + n3) / 3;
cout << "The average is " << average << endl;
return 0;
}
An Overview of C++
16
1.4 Expressions
Whenever you want a program to perform calculations, you need to write an expression
that specifies the necessary operations in a form similar to that used for expressions in
mathematics. For example, suppose that you wanted to solve the quadratic equation
ax2 + bx + c = 0
As you know from high-school mathematics, this equation has two solutions given by the
formula
x =
b
b2 4ac
2a
The first solution is obtained by using + in place of the symbol the second is obtained
by using instead. In C++, you could compute the first of these solutions by writing the
following expression:
(-b + sqrt(b * b - 4 * a * c)) / (2 * a)
An Overview of C++
17
Operator
()
[]
->
.
unary operators: - ++
*
/
%
+
<<
>>
<
<=
>
>=
==
!=
&
^
|
&&
||
?:
=
op=
,
--
&
(type)
sizeof
Associativity
left
right
left
left
left
left
left
left
left
left
left
left
right
right
left
operand, the one that appears higher in the precedence table is applied first. Thus, in the
expression
(-b + sqrt(b * b - 4 * a * c)) / (2 * a)
Should be (2 * a)
Without the parentheses, the division operator would be performed first because / and *
have the same precedence and associate to the left.
Mixing types in an expression
In C++, you can write an expression that includes values of different numeric types. If
C++ encounters an operator whose operands are of different numeric types, the compiler
automatically converts the operands to a common type by determining which of the two
An Overview of C++
18
operand types appears closest to the top in Table 1-5. The result of applying the
operation is always that of the arguments after any conversions are applied. This
convention ensures that the result of the computation is as precise as possible.
As an example, suppose that n is declared as an int, and x is declared as a double.
The expression
n + 1
is evaluated using integer arithmetic and produces a result of type int. The expression
x + 1
however, is evaluated by converting the integer 1 to the floating-point value 1.0 and
adding the results together using double-precision floating-point arithmetic, which results
in a value of type double.
Integer division and the remainder operator
The fact that applying an operator to two integer operands generates an integer result
leads to an interesting situation with respect to the division operator. If you write an
expression like
9 / 4
C++s rules specify that the result of this operation must be an integer, because both
operands are of type int . When C++ evaluates this expression, it divides 9 by 4 and
discards any remainder. Thus, the value of this expression in C++ is 2, not 2.25.
If you want to compute the mathematically correct result of 9 divided by 4, at least one
of the operands must be a floating-point number. For example, the three expressions
9.0 / 4
9 / 4.0
9.0 / 4.0
each produce the floating-point value 2.25. The decimal fraction is thrown away only if
both operands are of type int. The operation of discarding a decimal fraction is called
truncation.
There is an additional arithmetic operator that computes a remainder, which is
indicated in C++ by the percent sign (%). The % operator returns the remainder when the
first operand is divided by the second, and requires that both operands be of one of the
integer types. For example, the value of
Table 1-5 Type conversion hierarchy for numeric types
long double
double
float
unsigned long
long
unsigned int
int
unsigned short
short
char
most precise
least precise
An Overview of C++
19
9 % 4
is 1, since 4 goes into 9 twice, with 1 left over. The following are some other examples
of the % operator:
0 % 4
1 % 4
4 % 4
= 0
= 1
= 0
19 % 4 =
20 % 4 =
2001 % 4 =
3
0
1
The first step in evaluating the expression is to convert num to a double, after which the
division is performed using floating-point arithmetic as described in the section on
Mixing types in an expression earlier in this chapter.
As long as the conversion moves upward in the hierarchy shown in Table 1-5, the
conversion causes no loss of information. If, however, you convert a value of a more
precise type to a less precise one, some information may be lost. For example, if you use
a type cast to convert a value of type double to type int, any decimal fraction is simply
dropped. Thus, the value of the expression
int(1.9999)
is the integer 1.
The assignment operator
In C++, assignment of values to variables is built into the expression structure. The =
operator takes two operands, just like + or *. The left operand must indicate a value that
can change, which is typically a variable name. When the assignment operator is
executed, the expression on the right-hand side is evaluated, and the resulting value is
then stored in the variable that appears on the left-hand side. Thus, if you evaluate an
expression like
result = 1
1 You also may have seen the older C-style typecast where the typename is enclosed in parentheses, e.g.
quotient = (double)num / den. Both versions of the syntax have the same meaning.
An Overview of C++
20
the effect is that the value 1 is assigned to the variable result. In most cases, assignment
expressions of this sort appear in the context of simple statements, which are formed by
adding a semicolon after the expression, as in the line
result = 1;
that appears in the powertab.cpp program. Such statements are often called assignment
statements, although they in fact have no special status in the language definition.
The assignment operator converts the type of the value on the right-hand side so that it
matches the declared type of the variable. Thus, if the variable total is declared to be of
type double, and you write the assignment statement
total = 0;
the integer 0 is converted into a double as part of making the assignment. If n is declared
to be of type int, the assignment
n = 3.14159265;
has the effect of setting n to 3, because the value is truncated to fit in the integer variable.
Even though assignment operators usually occur in the context of simple statements,
they can also be incorporated into larger expressions, in which case the result of applying
the assignment operator is simply the value assigned. For example, the expression
z = (x = 6) + (y = 7)
has the effect of setting x to 6, y to 7, and z to 13. The parentheses are required in this
example because the = operator has a lower precedence than +. Assignments that are
written as part of larger expressions are called embedded assignments.
Although there are contexts in which embedded assignments are extremely convenient,
they often make programs more difficult to read because the assignment is easily
overlooked in the middle of a complex expression. For this reason, this text limits the use
of embedded assignments to a few special circumstances in which they seem to make the
most sense. Of these, the most important is when you want to set several variables to the
same value. C++s definition of assignment as an operator makes it possible, instead of
writing separate assignment statements, to write a single statement like
n1 = n2 = n3 = 0;
which has the effect of setting all three variables to 0. This statement works because C++
evaluates assignment operators from right to left. The entire statement is therefore
equivalent to
n1 = (n2 = (n3 = 0));
The expression n3 = 0 is evaluated, which sets n3 to 0 and then passes 0 along as the
value of the assignment expression. That value is assigned to n2, and the result is then
assigned to n1. Statements of this sort are called multiple assignments.
As a programming convenience, C++ allows you to combine assignment with a binary
operator to produce a form called a shorthand assignment. For any binary operator op,
the statement
variable op= expression;
An Overview of C++
21
is equivalent to
variable = variable op (expression);
where the parentheses are required only if the expression contains an operator whose
precedence is lower than that of op. Thus, the statement
balance += deposit;
is a shorthand for
balance = balance + deposit;
Similarly,
y--;
or
y = y - 1;
As it happens, these operators are more intricate than the previous examples would
suggest. To begin with, each of these operators can be written in two ways. The operator
An Overview of C++
22
The first form, in which the operator follows the operand, is called the postfix form, the
second, the prefix form.
If all you do is execute the ++ operator in isolationas you do in the context of a
separate statement or a typical for loop like those in the powertab.cpp examplethe
prefix and postfix operators have precisely the same effect. You notice the difference
only if you use these operators as part of a larger expression. Then, like all operators, the
++ operator returns a value, but the value depends on where the operator is written
relative to the operand. The two cases are as follows:
x++
Calculates the value of x first, and then increments it. The value
returned to the surrounding expression is the original value before
the increment operation is performed.
++x
Increments the value of x first, and then uses the new value as the
value of the ++ operation as a whole.
The -- operator behaves similarly, except that the value is decremented rather than
incremented.
You may wonder why would anyone use such an arcane feature. The ++ and - operators are certainly not essential. Moreover, there are not many circumstances in
which programs that embed these operators in larger expressions are demonstrably better
than those that use a simpler approach. On the other hand, ++ and -- are firmly
entrenched in the historical tradition shared by C++ programmers. They are idioms, and
programmers use them frequently. Because these operators are so common, you need to
understand them so that you can make sense of existing code.
Boolean operators
C++ defines three classes of operators that manipulate Boolean data: the relational
operators, the logical operators, and the ?: operator. The ;relational operators are used
to compare two values. C++ defines six relational operators, as follows:
==
!=
>
<
>=
<=
Equal
Not equal
Greater than
Less than
Greater than or equal to
Less than or equal to
When you write programs that test for equality, be careful to use the
operator, which is composed of two equal signs. A single equal
sign is the assignment operator. Since the double equal sign violates
conventional mathematical usage, replacing it with a single equal sign
is a particularly common mistake. This mistake can also be very
difficult to track down because the C++ compiler does not usually
catch it as an error. A single equal sign usually turns the expression
==
COMMON PITFALLS
When writing programs that
test for equality, be sure to
use the == operator and
not the single = operator,
which signifies assignment.
This error is extremely
common and can lead to
bugs that are very difficult
to find, because the
compiler cannot detect the
error.
An Overview of C++
23
into an embedded assignment, which is perfectly legal in C++; it just isnt at all what you
want.
The relational operators can be used to compare atomic data values like integers,
floating-point numbers, Boolean values, and characters. Some of the types supplied in
the libraries, such as string, also can be compared using the relational operators.
In addition to the relational operators, C++ defines three logical operators that take
Boolean operands and combine them to form other Boolean values:
!
&&
||
q
false
true
false
true
p && q
false
false
false
true
p || q
false
true
true
true
!p
true
true
false
false
C++ interprets the && and || operators in a way that differs from the interpretation
used in many other programming languages such as Pascal. Whenever a C++ program
evaluates an expression of the form
exp1 && exp2
or
exp1 || exp2
the individual subexpressions are always evaluated from left to right, and evaluation ends
as soon as the answer can be determined. For example, if exp1 is false in the expression
involving && , there is no need to evaluate exp2 since the final answer will always be
false . Similarly, in the example using || , there is no need to evaluate the second
operand if the first operand is true. This style of evaluation, which stops as soon as the
answer is known, is called short-circuit evaluation.
The C++ programming language provides another Boolean operator that can be
extremely useful in certain situations: the ?: operator. (This operator is referred to as
question-mark colon, even though the two characters do not appear adjacent to one
another in the code.) Unlike any other operator in C++, ?: is written in two parts and
requires three operands. The general form of the operation is
(condition) ? exp1 : exp2
An Overview of C++
24
The parentheses around the condition are not technically required, but C++ programmers
often include them to emphasize the boundaries of the conditional test.
When a C++ program encounters the ?: operator, it first evaluates the condition. If the
condition turns out to be true , exp1 is evaluated and used as the value of the entire
expression if the condition is false , the value is the result of evaluating exp2 . For
example, you can use the ?: operator to assign to max either the value of x or the value of
y, whichever is greater, as follows:
max = (x > y) ? x : y;
1.5 Statements
Programs in C++ are composed of functions, which are made up in turn of statements.
As in most languages, statements in C++ fall into one of two principal classifications:
simple statements, which perform some action, and control statements, which affect
the way in which other statements are executed. The sections that follow review the
principal statement forms available in C++ and give you the fundamental tools you need
to write your own programs.
Simple statements
The most common statement in C++ is the simple statement, which consists of an
expression followed by a semicolon:
expression;
When the C++ compiler encounters a block, it treats the entire block as a single
statement. Thus, whenever the notation statement appears in a pattern for one of the
control forms, you can substitute for it either a single statement or a block. To emphasize
that they are statements as far as the compiler is concerned, blocks are sometimes referred
to as compound statements. In C++, the statements in any block may be preceded by
declarations of variables. In this text, variable declarations are introduced only in the
block that defines the body of a function.
The statements in the interior of a block are usually indented relative to the enclosing
context. The compiler ignores the indentation, but the visual effect is extremely helpful
to the human reader, because it makes the structure of the program jump out at you from
An Overview of C++
25
the format of the page. Empirical research has shown that indenting three or four spaces
at each new level makes the program structure easiest to see; the programs in this text use
four spaces for each new level. Indentation is critical to good programming, so you
should strive to develop a consistent indentation style in your programs.
The only aspect of blocks that tends to cause any confusion for new students is the role
of the semicolon. In C++, the semicolon is part of the syntax of a simple statement; it
acts as a statement terminator rather than as a statement separator. While this rule is
perfectly consistent, it can cause trouble for people who have previously been exposed to
the language Pascal, which uses a different rule. In practical terms, the differences are:
1. In C++, there is always a semicolon at the end of the last simple statement in a block.
In Pascal, the semicolon is usually not present although most compilers allow it as an
option.
2. In C++, there is never a semicolon after the closing brace of a statement block. In
Pascal, a semicolon may or may not follow the END keyword, depending on context.
The convention for using semicolons in C++ has advantages for program maintenance
and should not cause any problem once you are used to it.
The if statement
In writing a program, you will often want to check whether some condition applies and
use the result of that check to control the subsequent execution of the program. This type
of program control is called conditional execution. The easiest way to express
conditional execution in C++ is by using the if statement, which comes in two forms:
if (condition) statement
if (condition) statement else statement
You use the first form of the if statement when your solution strategy calls for a set of
statements to be executed only if a particular Boolean condition is true. If the condition
is false, the statements that form the body of the if statement are simply skipped. You
use the second form of the if statement for situations in which the program must choose
between two independent sets of actions based on the result of a test. This statement
form is illustrated by the following program, which reads in a number and classifies it as
either even or odd.
int main() {
int n;
cout << "This program classifies a num even or odd." << endl;
cout << "Enter a number: ";
n = GetInteger();
if (n % 2 == 0) {
cout << "That number is even." << endl;
} else {
cout << "That number is odd." << endl;
}
return 0;
}
As with any control statement, the statements controlled by the if statement can be
either a single statement or a block. Even if the body of a control form is a single
An Overview of C++
26
statement, you are free to enclose it in a block if you decide that doing so improves the
readability of your code. The programs in this book enclose the body of every control
statement in a block unless the entire statementboth the control form and its bodyis
so short that it fits on a single line.
The switch statement
The if statement is ideal for those applications in which the program logic calls for a
two-way decision point: some condition is either true or false, and the program acts
accordingly. Some applications, however, call for more complicated decision structures
involving several mutually exclusive cases: in one case, the program should do x; in
another case, it should do y; in a third, it should do z; and so forth. In many applications,
the most appropriate statement to use for such situations is the switch statement, which
has the following syntactic form:
switch (e) {
case c1:
statements
break;
case c2:
statements
break;
. . . more case clauses
default:
statements
break;
}
. . .
The expression e is called the control expression. When the program executes a
it evaluates the control expression and compares it against the values
each of which must be a constant. If one of the constants matches the
value of the control expression, the statements in the associated case clause are executed.
When the program reaches the break statement at the end of the clause, the operations
specified by that clause are complete, and the program continues with the statement that
follows the entire switch statement.
switch statement,
c1, c 2 , and so forth,
The default clause is used to specify what action occurs if none of the constants
match the value of the control expression. The default clause, however, is optional. If
none of the cases match and there is no default clause, the program simply continues on
with the next statement after the switch statement without taking any action at all. To
avoid the possibility that the program might ignore an unexpected case, it is good
programming practice to include a default clause in every switch statement unless you
are certain you have enumerated all the possibilities, even if the default clause is simply
default:
Error("Unexpected case value");
The Error function is part of the genlib library and provides a uniform way of
responding to errors. This function takes one string parameter, the error message. The
Error function does not return; after the error message is displayed, the program
terminates.
The code pattern Ive used to illustrate the syntax of the switch statement deliberately
suggests that break statements are required at the end of each clause. In fact, C++ is
defined so that if the break statement is missing, the program starts executing statements
An Overview of C++
27
from the next clause after it finishes the selected one. While this design can be useful in
some cases, it causes many more problems than it solves. To reinforce the importance of
remembering to exit at the end of each case clause, the programs in this text always
include a break or return statement in each such clause.
The one exception to this rule is that multiple case lines specifying
different constants can appear together, one after another, before the
same statement group. For example, a switch statement might
include the following code:
case 1:
case 2:
statements
break;
COMMON PITFALLS
It is good programming
practice to include a break
statement at the end of
every case clause within a
switch statement. Doing
so will help you avoid
programming errors that
can be extremely difficult to
find. It is also good
practice to include a
default clause unless
you are sure you have
covered all the cases.
The operation of the switch statement is illustrated by the following function, which
computes the number of days for a given month and year:
int MonthDays(int month, int year) {
switch (month) {
case September:
case April:
case June:
case November:
return 30;
case February:
return (IsLeapYear(year)) ? 29 : 28;
default:
return 31;
}
}
The code assumes that there is a function IsLeapYear(year) which tests whether year
is a leap year and that the names of the months have been defined using constants, as
follows:
const
const
const
const
const
const
const
const
const
const
const
const
int
int
int
int
int
int
int
int
int
int
int
int
JANUARY
= 1;
FEBRUARY = 2;
MARCH
= 3;
APRIL
= 4;
MAY
= 5;
JUNE
= 6;
JULY
= 7;
AUGUST
= 8;
SEPTEMBER = 9;
OCTOBER = 10;
NOVEMBER = 11;
DECEMBER = 12;
An Overview of C++
28
The constants in a switch statement must be of integer type or a type that behaves like
an integer. (The actual restriction is that the type must be a scalar type, which is defined
in Chapter 2.) In particular, characters are often used as case constants, as illustrated by
the following function, which tests to see if its argument is a vowel:
bool IsVowel(char ch) {
switch (ch) {
case 'A': case 'E': case 'I': case 'O': case 'U':
case 'a': case 'e': case 'i': case 'o': case 'u':
return true;
default:
return false;
}
}
An Overview of C++
29
line itself seems to introduce an infinite loop because the value of the constant true can
never become false. The only way this program can exit from the loop is by executing
the break statement inside it. The loop-and-a-half strategy is illustrated by the
addlist.cpp program in Figure 1-4, which computes the sum of a list of integers
terminated by the sentinel value 0.
An Overview of C++
30
There are other strategies for solving the loop-and-a-half problem that involve copying
part of the code outside the loop. However, empirical studies have demonstrated that
students are more likely to write correct programs if they use a break statement to exit
from the middle of the loop than if they are forced to use some other strategy. This
evidence and my own experience have convinced me that using the break statement
inside a while loop is the best solution to the loop-and-a-half problem.
The for statement
One of the most important control statements in C++ is the for statement, which is used
in situations in which you want to repeat an operation a particular number of times. The
general form is
for (init; test; step) {
statements
}
An Overview of C++
31
The operation of the for loop is determined by the three italicized expressions on the
control line: init, test, and step. The init expression indicates how the for loop should
be initialized and usually declares and initializes the index variable. For example, if you
write
for
for (int i = 0; . . .
the loop will begin by setting the index variable i to 0. If the loop begins
for (int i = -7; . . .
begins with i equal to 0 and continues as long as i is less than n , which turns out to
represent a total of n cycles, with i taking on the values 0, 1, 2, and so forth, up to the
final value n1. The loop
for (int i = 1; i <= n; i++)
begins with i equal to 1 and continues as long as i is less than or equal to n. This loop
also runs for n cycles, with i taking on the values 1, 2, and so forth, up to n.
The step expression indicates how the value of the index variable changes from cycle to
cycle. The most common form of step specification is to increment the index variable
using the ++ operator, but this is not the only possibility. For example, one can count
backward by using the -- operator, or count by twos by using += 2 instead of ++.
As an illustration of counting in the reverse direction, the program
int main() {
for (int t = 10; t >= 0; t--) {
cout << t << endl;
}
cout << "Liftoff!" << endl;
return 0;
}
2 The C++ standard states that the scope of an index variable declared in the initialization expression of a
for loop extends to the end of the loop body and no further. Some C++ compilers, such as Visual C++ 6.0,
mistakenly extend the scope to the end of the block enclosing the entire loop.
An Overview of C++
32
The expressions init, test, and s t e p in a for statement are each optional, but the
semicolons must appear. If init is missing, no initialization is performed. If test is
missing, it is assumed to be true . If step is missing, no action occurs between loop
cycles.
1.6 Functions
A function consists of a set of statements that have been collected together and given a
name. The act of using the name to invoke the associated statements is known as calling
that function. To indicate a function call in C++, you write the name of the function,
followed by a list of expressions enclosed in parentheses. These expressions, called
arguments, allow the calling program to pass information to the function. For example,
in the p o w e r t a b . c p p program at the beginning of this chapter, the function
RaiseIntToPower took two integer arguments, n and k, which are the values it needs to
know in order to compute nk. If a function requires no information from its caller, it need
not have any arguments, but an empty set of parentheses must appear in the function call.
Once called, the function takes the data supplied as arguments, does its work, and then
returns to the program step from which the call was made. Remembering what the
calling program was doing and being able to get back precisely to that point is one of the
defining characteristics of the function-calling mechanism. The operation of going back
to the calling program is called returning from the function.
Returning results from functions
As they return, functions can send results back to the calling program. Thus, when the
RaiseIntToPower function returns with the statement
return result;
the value of the local variable result is passed back to the main program as the value of
the function. This operation is called returning a value.
Functions can return values of any type. The following function, for example, returns
a value of type bool, which can then be used in conditional expressions:
bool IsLeapYear(int year) {
return ((year % 4 == 0) && (year % 100 != 0))
|| (year % 400 == 0);
}
An Overview of C++
33
Functions that return Boolean results play an important role in programming and are
called predicate functions.
Functions, however, do not need to return a value at all. A function that does not
return a value and is instead executed for its effect is called a procedure. Procedures are
indicated in the definition of a function by using the reserved word void as the result
type.
The return statement in C++ has two forms. For procedures, you write the statement
as
return;
For functions that return a value, the return keyword is followed by an expression, as
follows:
return expression;
Executing either form of the return statement causes the current function to return
immediately to its caller, passing back the value of the expression, if any, to its caller as
the value of the function.
Function definitions and prototypes
A function definition has the following syntactic form:
result-type name(parameter-list) {
. . . body . . .
}
In this example, result-type is the type of value returned by the function, name is the
function name, and parameter-list is a list of declarations separated by commas, giving the
type and name of each parameter to the function. Parameters are placeholders for the
arguments supplied in the function call and act like local variables except for the fact that
they are given initial values by the calling program. If a function takes no parameters, the
entire parameter list in the function header line is empty. The body of the function is a
block and typically contains declarations for the local variables required by the function.
Before you use a function in a C++ program, you declare it by specifying its prototype.
A prototype has exactly the same form as a function definition, except that the entire
body is replaced by a semicolon. The names of the parameter variables are optional in a
prototype, but supplying them usually helps the reader.
The mechanics of the function-calling process
When you call a function in a program, the following steps occur:
1. The calling program computes values for each argument. Because the arguments are
expressions, this computation can involve operators and other functions, all of which
are evaluated before execution of the new function actually begins.
2. The system creates new space for all the local variables required by the new function,
including the parameter variables. These variables are usually allocated together in a
block, which is called a stack frame.
3. The value of each argument is copied into the corresponding parameter variable. If
there is more than one argument, the arguments are copied into the parameters in
An Overview of C++
4.
5.
6.
7.
34
order; the first argument is copied into the first parameter, and so forth. If necessary,
type conversions are performed between the argument values and the parameter
variables, as in an assignment statement. For example, if you pass a value of type int
to a function that expects a parameter of type double, the integer is converted into the
equivalent floating-point value before it is copied into the parameter variable.
The statements in the function body are executed until a return statement is
encountered or there are no more statements to execute.
The value of the return expression, if any, is evaluated and returned as the value of
the function. If the value being returned does not precisely match the result type
declared for the function, a type conversion is performed. Thus, if a return
statement specifies a floating-point value in a function defined to return an int, the
result is truncated to an integer.
The stack frame created for this function call is discarded. In the process, all local
variables disappear.
The calling program continues, with the returned value substituted in place of the call.
inside the function sets the local copy to 0 but leaves x unchanged in the calling program.
To address this problem, you can change the parameter to a reference parameter by
adding an ampersand to the parameter declaration in the function header. Now the
parameter value will not be copied, instead a reference is made to the original variable.
Changes to the parameter are reflected in the original variable. The new coding is
void SetToZero(int & var) {
var = 0;
}
To use this function, the caller must pass an assignable integer variable. To set x to 0,
for example, you would need to make the following call:
SetToZero(x);
An Overview of C++
35
The use of reference parameters makes it possible for functions to change values in the
frame of their caller. This mechanism is referred to as call by reference.
In C++, one of the common uses of call by reference occurs when a function needs to
return more than one value to the calling program. A single result can easily be returned
as the value of the function itself. If you need to return more than one result from a
function, the return value is no longer appropriate. The standard approach to solving the
problem is to turn that function into a procedure and pass values back and forth through
the argument list.
As an example, suppose that you wanted to write a program to solve the quadratic
equation
ax2 + bx + c = 0
but thatbecause of your desire to practice good programming styleyou were
committed to dividing the work of the program into three phases as represented by the
boxes in the following flowchart:
Input phase:
Accept values of
the coefficients
from the user.
Computation phase:
Solve the quadratic
equation for those
coefficients.
Output phase:
Display the roots
of the equation
on the screen.
Decomposing this problem into separate functions that are responsible for each of these
phases is somewhat tricky because several values must be passed from each phase to the
next. Because there are three coefficients, you would like the input phase to return three
values. Similarly, the computation phase must return two values, because a quadratic
equation has two solutions.
Figure 1-5 shows how call by reference makes it possible to decompose the quadratic
equation problem in this way. At each level, parameters that act as input to each function
are passed in the conventional way; parameters that represent output from the function
are passed by reference.
Stepwise refinement
Procedures and functions enable you to divide a large programming problem into smaller
pieces that are individually easy to understand. The process of dividing a problem into
manageable pieces, called decomposition, is a fundamental programming strategy.
Finding the right decomposition, however, turns out to be a difficult task that requires
considerable practice. If you choose the individual pieces well, each one will have
An Overview of C++
36
<iostream>
<cmath>
"genlib.h"
"simpio.h"
An Overview of C++
37
/*
* Function: SolveQuadratic
* Usage: SolveQuadratic(a, b, c, x1, x2);
* --------------------------------------* This function solves a quadratic equation. The coefficients
* are supplied as the arguments a, b, and c, and the roots are
* returned in x1 and x2, which are reference parameters.
*/
void SolveQuadratic(double a, double b, double c,
double & x1, double & x2) {
if (a == 0) Error("The coefficient a must be nonzero");
double disc = b * b - 4 * a * c;
if (disc < 0) Error("The solutions are complex numbers");
double sqrtDisc = sqrt(disc);
x1 = (-b + sqrtDisc) / (2 * a);
x2 = (-b - sqrtDisc) / (2 * a);
}
/*
* Function: DisplayRoots
* Usage: DisplayRoots(x1, x2);
* ---------------------------* This function displays the values x1 and x2, which are
* the roots of a quadratic equation.
*/
void DisplayRoots(double x1, double x2) {
if (x1 == x2) {
cout << "There is a double root at " << x1 << endl;
} else {
cout << "The roots are " << x1 << " and " << x2 << endl;
}
}
conceptual integrity as a unit and make the program as a whole much simpler to
understand. But if you choose unwisely, the decomposition can get in your way. There
are no hard-and-fast rules for selecting a particular decomposition; you will learn how to
apply this process through experience.
When you are faced with the task of writing a program, the best strategy is usually to
start with the main program. At this level, you think about the problem as a whole and
then try to identify the major pieces of the entire task. Once you figure out what the big
pieces of the program are, you can define them as independent functions. Since some of
these functions may themselves be complicated, it is often appropriate to decompose
them into still smaller ones. You can continue this process until every piece of the
problem is simple enough to be solved on its own. This process is called top-down
design, or stepwise refinement.
Summary
This chapter is itself a summary, which makes it hard to condense it to a few central
points. Its purpose was to introduce you to the C++ programming language and give you
a crash course in how to write simple programs in that language. This chapter
concentrated on the low-level structure of the language, proceeding in turn through the
An Overview of C++
38
topics of expressions, statements, and functions. The facilities that C++ offers for
defining new data structures are detailed in Chapter 2.
Important points in this chapter include:
In the 25 years of its existence, the C++ programming language has become one of the
most widely used languages in the world.
A typical C++ program consists of comments, library inclusions, program-level
definitions, function prototypes, a function named main that is called when the
program is started, and a set of auxiliary function definitions that work together with
the main program to accomplish the required task.
Variables in a C++ program must be declared before they are used. Most variables in
C++ are local variables, which are declared within a function and can only be used
inside the body of that function.
A data type is defined by a domain of values and a set of operations. C++ includes
several predefined types that allow programs to store data of several different types,
such as integers, floating-point numbers, Booleans, and characters. In addition to these
built-in types, the standard library defines the type string, which is treated in this
book as if it were an integral part of the language.
The easiest way to read input data from the user is to call functions in the simplified
I/O library (simpio ), which defines such functions as GetInteger, GetReal, and
GetLine. To display output on the computer screen, the usual approach is to insert the
values into the standard cout stream using the insertion operator <<.
Expressions in C++ are written in a form similar to that in most programming
languages, with individual terms connected by operators. A complete list of the
operators available in C++ appears in Table 1-4, which also indicates the relative
precedence of each operator.
Statements in C++ fall into two classes: simple statements and control statements. A
simple statement consists of an expressiontypically an assignment or a function
callfollowed by a semicolon. The control statements described in this chapter are
the i f , switch , while , and for statements. The first two are used to express
conditional execution, while the last two are used to specify repetition.
C++ programs are typically subdivided into several functions. Each function consists
of a sequence of statements that can be invoked by writing the name of the function,
followed by a list of arguments enclosed in parentheses. These arguments are copied
into the corresponding parameter variables inside the function. The function can
return a result to the caller by using the return statement and can share values using
reference parameters.
Review questions
1.
2.
3.
In an #include line, the name of the library header file can be enclosed in either
angle brackets of double quotation marks. What is the difference between the two
forms of punctuation?
4.
How would you define a constant called CENTIMETERS_PER_INCH with the value
2.54?
An Overview of C++
5.
What is the name of the function that must be defined in every C++ program?
6.
What is the purpose of inserting endl into the output stream cout?
7.
8.
9.
39
10. What is the difference between the types short, int, and long?
11. What does ASCII stand for?
12. List all possible values of type bool.
13. What statements would you include in a program to read a value from the user and
store it in the variable x, which is declared as a double?
14. Suppose that a function contains the following declarations:
int i;
double d;
char c;
string s;
Write a statement that displays the values of each of these variables on the screen.
15. Indicate the values and types of the following expressions:
a. 2 + 3
d. 3 * 6.0
b. 19 / 5
e. 19 % 5
c. 19.0 / 5
f. 2 % 7
16. What is the difference between the unary minus and the binary subtraction operator?
17. What does the term truncation mean?
18. Calculate the result of each of the following expressions:
a. 6 + 5 / 4 - 3
b. 2 + 2 * (2 * 2 - 2) % 2 / 2
c. 10 + 9 * ((8 + 7) % 6) + 5 * 4 % 3 * 2 + 1
d. 1 + 2 + (3 + 4) * ((5 * 6 % 7 * 8) - 9) - 10
19. How do you specify a shorthand assignment operation?
20. What is the difference between the expressions ++x and x++?
21. What does the term short-circuit evaluation mean?
An Overview of C++
40
22. Write out the general syntactic form for each of the following control statements: if,
switch, while, for.
23. Describe in English the general operation of the switch statement.
24. What is a sentinel?
25. What for loop control line would you use in each of the following situations?
a. Counting from 1 to 100
b. Counting by sevens starting at 0 until the number has more than two digits
c. Counting backward by twos from 100 to 0
26. What is a function prototype?
27. In your own words, describe what happens when you call a function in C++.
28. What is meant by the term stepwise refinement?
Programming exercises
1.
Write a program that reads in a temperature in degrees Celsius and displays the
corresponding temperature in degrees Fahrenheit. The conversion formula is
9
F = 5 C + 32
2.
3.
4.
Write a program that reads in a positive integer N and then calculates and displays
the sum of the first N odd integers. For example, if N is 4, your program should
display the value 16, which is 1 + 3 + 5 + 7.
5.
Write a program that reads in a list of integers from the user until the user enters the
value 0 as a sentinel. When the sentinel appears, your program should display the
largest value in the list, as illustrated in the following sample run:
This program finds the largest integer in a list.
Enter 0 to signal the end of the list.
? 17
? 42
? 11
? 19
? 35
? 0
The largest value is 42
An Overview of C++
6.
41
Using the DigitSum function from the section entitled The while statement as a
model, write a program that reads in an integer and then displays the number that has
the same digits in the reverse order, as illustrated by this sample run:
This program reverses the digits in an integer.
Enter a positive integer: 123456789
The reversed number is 987654321
To make sure your program can handle integers as large as the one shown in the
example, use the type long instead of int in your program.
7.
Greek mathematicians took a special interest in numbers that are equal to the sum of
their proper divisors (a proper divisor of N is any divisor less than N itself). They
called such numbers perfect numbers. For example, 6 is a perfect number because
it is the sum of 1, 2, and 3, which are the integers less than 6 that divide evenly into
6. Similarly, 28 is a perfect number because it is the sum of 1, 2, 4, 7, and 14.
Write a predicate function IsPerfect that takes an integer n and returns true if n
is perfect, and false otherwise. Test your implementation by writing a main
program that uses the IsPerfect function to check for perfect numbers in the range
1 to 9999 by testing each number in turn. When a perfect number is found, your
program should display it on the screen. The first two lines of output should be 6
and 28. Your program should find two other perfect numbers in the range as well.
8.
9.
10. The German mathematician Leibniz (16461716) discovered the rather remarkable
fact that the mathematical constant can be computed using the following
mathematical relationship:
An Overview of C++
42
1 + 1 1 + 1 1 + . . .
4 1
3
5
7
9
11
The formula to the right of the equal sign represents an infinite series; each fraction
represents a term in that series. If you start with 1, subtract one-third, add one-fifth,
and so on, for each of the odd integers, you get a number that gets closer and closer
to the value of /4 as you go along.
Write a program that calculates an approximation of consisting of the first
10,000 terms in Leibnizs series.
11. You can also approximate by approximating the area bounded by a circular arc.
Consider the following quarter circle:
which has a radius r equal to two inches. From the formula for the area of a circle,
you can easily determine that the area of the quarter circle should be square inches.
You can also approximate the area computationally by adding up the areas of a series
of rectangles, where each rectangle has a fixed width and the height is chosen so that
the circle passes through the midpoint of the top of the rectangle. For example, if
you divide the area into 10 rectangles from left to right, you get the following
diagram:
The sum of the areas of the rectangles approximates the area of the quarter circle.
The more rectangles there are, the closer the approximation.
For each rectangle, the width w is a constant derived by dividing the radius by the
number of rectangles. The height h, on the other hand, varies depending on the
position of the rectangle. If the midpoint of the rectangle in the horizontal direction
is given by x, the height of the rectangle can be computed using the distance formula
h =
r 2 x2
An Overview of C++
43
Write a program to compute the area of the quarter circle by dividing it into 100
rectangles.
12. When you write a check, the dollar amount appears twice: once as a number and once
as English text. For example, if you write a check for $1729, you need to translate
that number to the English text one thousand seven hundred twenty-nine. Your task
in this problem is to write a program that reads in integers from the user and writes
out the equivalent value in figures on the next line, stopping when the user enters any
negative number. For example, the following is a sample run of this program:
Enter numbers in figures; use a negative value to stop.
Number: 0
zero
Number: 1
one
Number: 11
eleven
Number: 256
two hundred fifty-six
Number: 1729
one thousand seven hundred twenty-nine
Number: 2001
two thousand one
Number: 12345
twelve thousand three hundred forty-five
Number: 13000
thirteen thousand
Number: -1
The key idea in this exercise is decomposition. The problem is not nearly as hard as
it looks if you break it down into separate procedures that accomplish parts of the
task. Many of these procedures will have a form that looks something like this:
void PrintOneDigit(int d) {
switch (d) {
case 0: cout << "zero"; break;
case 1: cout << "one"; break;
case 2: cout << "two"; break;
case 3: cout << "three"; break;
case 4: cout << "four"; break;
case 5: cout << "five"; break;
case 6: cout << "six"; break;
case 7: cout << "seven"; break;
case 8: cout << "eight"; break;
case 9: cout << "nine"; break;
default: Error("Illegal call to PrintOneDigit");
}
}
In writing your program, you should keep the following points in mind:
You dont need to perform any string manipulation. All you have to do is display
the value on the screen, which means that inserting to cout is all you need.
Your program need work only with values up to 999,999, although it should give
the user some kind of error message if a number is outside of its range.
An Overview of C++
44
It is perfectly acceptable for all the letters in the output to be lowercase. The
problem is much harder if you try to capitalize the first word.
You should remain on the lookout for functions that you can reuse. For example,
printing the number of thousands is pretty much the same as printing out the last
three digits, and you should be able to use the same procedure more than once.
Several special cases arise in this problem. For example, the number 11 must be
treated differently than 21 or 31, because eleven doesnt fit the pattern established
by twenty-one and thirty-one.