Suppose you want to save time and space by passing the address of a structure instead of passing the entire structure. This requires rewriting the functions so that they use pointers to structures. First, let’s look at how you rewrite the show_polar()
function. You need to make three changes:
• When calling the function, pass it the address of the structure (&pplace
) rather than the structure itself (pplace
).
• Declare the formal parameter to be a pointer-to-polar
—that is, type polar *
. Because the function shouldn’t modify the structure, use the const
modifier.
• Because the formal parameter is a pointer instead of a structure, use the indirect membership operator (->
) rather than the membership operator (dot).
After these changes are made, the function looks like this:
// show polar coordinates, converting angle to degrees
void show_polar (const polar * pda)
{
using namespace std;
const double Rad_to_deg = 57.29577951;
cout << "distance = " << pda->distance;
cout << ", angle = " << pda->angle * Rad_to_deg;
cout << " degrees
";
}
Next, let’s alter rect_to_polar
. This is more involved because the original rect_to_polar
function returns a structure. To take full advantage of pointer efficiency, you should use a pointer instead of a return value. The way to do this is to pass two pointers to the function. The first points to the structure to be converted, and the second points to the structure that’s to hold the conversion. Instead of returning a new structure, the function modifies an existing structure in the calling function. Hence, although the first argument is const
pointer, the second is not const
. Otherwise, you apply the same principles used to convert show_polar()
to pointer arguments. Listing 7.13 shows the reworked program.
// strctptr.cpp -- functions with pointer to structure arguments
#include <iostream>
#include <cmath>
// structure templates
struct polar
{
double distance; // distance from origin
double angle; // direction from origin
};
struct rect
{
double x; // horizontal distance from origin
double y; // vertical distance from origin
};
// prototypes
void rect_to_polar(const rect * pxy, polar * pda);
void show_polar (const polar * pda);
int main()
{
using namespace std;
rect rplace;
polar pplace;
cout << "Enter the x and y values: ";
while (cin >> rplace.x >> rplace.y)
{
rect_to_polar(&rplace, &pplace); // pass addresses
show_polar(&pplace); // pass address
cout << "Next two numbers (q to quit): ";
}
cout << "Done.
";
return 0;
}
// show polar coordinates, converting angle to degrees
void show_polar (const polar * pda)
{
using namespace std;
const double Rad_to_deg = 57.29577951;
cout << "distance = " << pda->distance;
cout << ", angle = " << pda->angle * Rad_to_deg;
cout << " degrees
";
}
// convert rectangular to polar coordinates
void rect_to_polar(const rect * pxy, polar * pda)
{
using namespace std;
pda->distance =
sqrt(pxy->x * pxy->x + pxy->y * pxy->y);
pda->angle = atan2(pxy->y, pxy->x);
}
Some compilers require explicit instructions to search the math library. For example, older versions of g++ use this command line:
g++ structfun.C -lm
From the user’s standpoint, the program in Listing 7.13 behaves like that in Listing 7.12. The hidden difference is that Listing 7.12 works with copies of structures, whereas Listing 7.13 uses pointers, allowing the functions to operate on the original structures.
string
Class ObjectsAlthough C-style strings and string
class objects serve much the same purpose, a string
class object is more closely related to a structure than to an array. For example, you can assign a structure to another structure and an object to another object. You can pass a structure as a complete entity to a function, and you can pass an object as a complete entity. If you need several strings, you can declare a one-dimensional array of string
objects instead of a two-dimensional array of char
.
Listing 7.14 provides a short example that declares an array of string objects and passes the array to a function that displays the contents.
// topfive.cpp -- handling an array of string objects
#include <iostream>
#include <string>
using namespace std;
const int SIZE = 5;
void display(const string sa[], int n);
int main()
{
string list[SIZE]; // an array holding 5 string object
cout << "Enter your " << SIZE << " favorite astronomical sights:
";
for (int i = 0; i < SIZE; i++)
{
cout << i + 1 << ": ";
getline(cin,list[i]);
}
cout << "Your list:
";
display(list, SIZE);
return 0;
}
void display(const string sa[], int n)
{
for (int i = 0; i < n; i++)
cout << i + 1 << ": " << sa[i] << endl;
}
Here’s a sample run of the program in Listing 7.14:
Enter your 5 favorite astronomical sights:
1: Orion Nebula
2: M13
3: Saturn
4: Jupiter
5: Moon
Your list:
1: Orion Nebula
2: M13
3: Saturn
4: Jupiter
5: Moon
The main point to note in this example is that, aside from the getline()
function, this program treats string
just as it would treat any of the built-in types, such as int
. If you want an array of string
, you just use the usual array-declaration format:
string list[SIZE]; // an array holding 5 string object
Each element of the list
array, then, is a string
object and can be used as such:
getline(cin,list[i]);
Similarly, the formal argument sa
is a pointer to a string
object, so sa[i]
is a string
object and can be used accordingly:
cout << i + 1 << ": " << sa[i] << endl;
array
ObjectsClass objects in C++ are based on structures, so some of the same programming considerations that apply to structures also apply to classes. For example, you can pass an object by value to a function, in which case the function acts on a copy of the original object. Alternatively, you can pass a pointer to an object, which allows the function to act on the original object. Let’s look at an example using the C++11 array
template class.
Suppose we have an array
object intended to hold expense figures for each of the four seasons of the year:
std::array<double, 4> expenses;
(Recall that using the array
class requires the array
header file and that the name array
is part of the std
namespace.) If we want a function to display the contents of expenses
, we can pass expenses
by value:
show(expenses);
But if we want a function that modifies the expenses
object, we need to pass the address of the object to the function:
fill(&expenses);
(The next chapter discusses an alternative approach, using references.) This is the same approach that Listing 7.13 used for structures.
How can we declare these two functions? The type of expenses
is array<double, 4>
, so that’s what must appear in the prototypes:
void show(std::array<double, 4> da); // da an object
void fill(std::array<double, 4> * pa); // pa a pointer to an object
These considerations form the core of the sample program. The program adds a few more features. First, it replaces 4
with a symbolic constant:
const int Seasons = 4;
Second, it adds a const array
object containing four string
objects representing the four seasons:
const std::array<std::string, Seasons> Snames =
{"Spring", "Summer", "Fall", "Winter"};
Note that the array
template is not limited to holding the basic data types; it can use class types too. Listing 7.15 presents the program in full.
//arrobj.cpp -- functions with array objects (C++11)
#include <iostream>
#include <array>
#include <string>
// constant data
const int Seasons = 4;
const std::array<std::string, Seasons> Snames =
{"Spring", "Summer", "Fall", "Winter"};
// function to modify array object
void fill(std::array<double, Seasons> * pa);
// function that uses array object without modifying it
void show(std::array<double, Seasons> da);
int main()
{
std::array<double, Seasons> expenses;
fill(&expenses);
show(expenses);
return 0;
}
void fill(std::array<double, Seasons> * pa)
{
using namespace std;
for (int i = 0; i < Seasons; i++)
{
cout << "Enter " << Snames[i] << " expenses: ";
cin >> (*pa)[i];
}
}
void show(std::array<double, Seasons> da)
{
using namespace std;
double total = 0.0;
cout << "
EXPENSES
";
for (int i = 0; i < Seasons; i++)
{
cout << Snames[i] << ": $" << da[i] << endl;
total += da[i];
}
cout << "Total Expenses: $" << total << endl;
}
Enter Spring expenses: 212
Enter Summer expenses: 256
Enter Fall expenses: 208
Enter Winter expenses: 244
EXPENSES
Spring: $212
Summer: $256
Fall: $208
Winter: $244
Total: $920
3.145.35.247