The usual C++ practice for passing class objects to a function is to use references. For instance, you would use reference parameters for functions taking objects of the string
, ostream
, istream
, ofstream
, and ifstream
classes as arguments.
Let’s look at an example that uses the string
class and illustrates some different design choices, some of them bad. The general idea is to create a function that adds a given string to each end of another string. Listing 8.7 provides three functions that are intended to do this. However, one of the designs is so flawed that it may cause the program to crash or even not compile.
// strquote.cpp -- different designs
#include <iostream>
#include <string>
using namespace std;
string version1(const string & s1, const string & s2);
const string & version2(string & s1, const string & s2); // has side effect
const string & version3(string & s1, const string & s2); // bad design
int main()
{
string input;
string copy;
string result;
cout << "Enter a string: ";
getline(cin, input);
copy = input;
cout << "Your string as entered: " << input << endl;
result = version1(input, "***");
cout << "Your string enhanced: " << result << endl;
cout << "Your original string: " << input << endl;
result = version2(input, "###");
cout << "Your string enhanced: " << result << endl;
cout << "Your original string: " << input << endl;
cout << "Resetting original string.
";
input = copy;
result = version3(input, "@@@");
cout << "Your string enhanced: " << result << endl;
cout << "Your original string: " << input << endl;
return 0;
}
string version1(const string & s1, const string & s2)
{
string temp;
temp = s2 + s1 + s2;
return temp;
}
const string & version2(string & s1, const string & s2) // has side effect
{
s1 = s2 + s1 + s2;
// safe to return reference passed to function
return s1;
}
const string & version3(string & s1, const string & s2) // bad design
{
string temp;
temp = s2 + s1 + s2;
// unsafe to return reference to local variable
return temp;
}
Here is a sample run of the program in Listing 8.7:
Enter a string: It's not my fault.
Your string as entered: It's not my fault.
Your string enhanced: ***It's not my fault.***
Your original string: It's not my fault.
Your string enhanced: ###It's not my fault.###
Your original string: ###It's not my fault.###
Resetting original string.
At this point the program crashed.
Version 1 of the function in Listing 8.7 is the most straightforward of the three:
string version1(const string & s1, const string & s2)
{
string temp;
temp = s2 + s1 + s2;
return temp;
}
It takes two string
arguments and uses string
class addition to create a new string that has the desired properties. Note that the two function arguments are const
references. The function would produce the same end result if it just passed string
objects:
string version4(string s1, string s2) // would work the same
In this case, s1
and s2
would be brand-new string
objects. Thus, using references is more efficient because the function doesn’t have to create new objects and copy data from the old objects to the new. The use of the const
qualifier indicates that this function will use, but not modify, the original strings.
The temp
object is a new object, local to the version1()
function, and it ceases to exist when the function terminates. Thus, returning temp
as a reference won’t work, so the function type is string
. This means the contents of temp
will be copied to a temporary return location. Then, in main()
, the contents of the return location are copied to the string named result
:
result = version1(input, "***");
The version2()
function doesn’t create a temporary string. Instead, it directly alters the original string:
const string & version2(string & s1, const string & s2) // has side effect
{
s1 = s2 + s1 + s2;
// safe to return reference passed to function
return s1;
}
This function is allowed to alter s1
because s1
, unlike s2
, is not declared using const
.
Because s1
is a reference to an object (input
) in main()
, it’s safe to return s1
as a reference. Because s1
is a reference to input
, the line
result = version2(input, "###");
essentially becomes equivalent to the following:
version2(input, "###"); // input altered directly by version2()
result = input; // reference to s1 is reference to input
However, because s1
is a reference to input
, calling this function has the side effect of altering input
also:
Your original string: It's not my fault.
Your string enhanced: ###It's not my fault.###
Your original string: ###It's not my fault.###
Thus, if you want to keep the original string unaltered, this is the wrong design.
The third version in Listing 8.7 is a reminder of what not to do:
const string & version3(string & s1, const string & s2) // bad design
{
string temp;
temp = s2 + s1 + s2;
// unsafe to return reference to local variable
return temp;
}
It has the fatal flaw of returning a reference to a variable declared locally inside version3()
. This function compiles (with a warning), but the program crashes when attempting to execute the function. Specifically, the following assignment aspect causes the problem:
result = version3(input, "@@@");
The program attempts to refer to memory that is no longer in use.
3.145.91.254