2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1...

33
2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3 Restrictions on Operator Overloading 8.4 Operator Functions as Class Members vs. Global Functions 8.5 Overloading Stream-Insertion and Stream-Extraction Operators 8.6 Overloading Unary Operators 8.7 Overloading Binary Operators 8.8 Converting between Types 8.9 Overloading ++ and -- 8.10 Case Study: A Date Class

Transcript of 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1...

Page 1: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

1

Chapter 8 - Operator Overloading

 

Outline8.1 Introduction8.2 Fundamentals of Operator Overloading8.3 Restrictions on Operator Overloading8.4 Operator Functions as Class Members vs. Global Functions8.5 Overloading Stream-Insertion and Stream-Extraction Operators8.6 Overloading Unary Operators8.7 Overloading Binary Operators8.8 Converting between Types8.9 Overloading ++ and --8.10 Case Study: A Date Class

Page 2: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

2

8.1 Introduction

• Use operators with objects (operator overloading)– Clearer than function calls for certain classes

– Operator sensitive to context

• Examples– <<

• Stream insertion, bitwise left-shift

– +• Performs arithmetic on multiple types (integers, floats, etc.)

• Will discuss when to use operator overloading

Page 3: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

3

8.2 Fundamentals of Operator Overloading

• Types– Built in (int, char) or user-defined

– Can use existing operators with user-defined types• Cannot create new operators

• Overloading operators– Create a function for the class

– Name function operator followed by symbol• Operator+ for the addition operator +

Page 4: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

4

8.2 Fundamentals of Operator Overloading

• Using operators on a class object– It must be overloaded for that class

• Exceptions:

– Assignment operator, =• Memberwise assignment between objects

– Address operator, &• Returns address of object

– Both can be overloaded

• Overloading provides concise notation– object2 = object1.add(object2);– object2 = object2 + object1;

Page 5: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

5

8.3 Restrictions on Operator Overloading

• Cannot change– How operators act on built-in data types

• I.e., cannot change integer addition

– Precedence of operator (order of evaluation)• Use parentheses to force order-of-operations

– Associativity (left-to-right or right-to-left)

– Number of operands• & is unitary, only acts on one operand

• Cannot create new operators• Operators must be overloaded explicitly

– Overloading + does not overload +=

Page 6: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

6

8.3 Restrictions on Operator Overloading

Operators that cannot be overloaded

. .* :: ?: sizeof

Operators that can be overloaded

+ - * / % ^ & |

~ ! = < > += -= *=

/= %= ^= &= |= << >> >>=

<<= == != <= >= && || ++

-- ->* , -> [] () new delete

new[] delete[]

Page 7: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

7

8.4 Operator Functions As Class Members Vs. Global Functions

• Operator functions– Member functions

• Use this keyword to implicitly get argument

• Gets left operand for binary operators (like +)

• Leftmost object must be of same class as operator

– Non member functions• Need parameters for both operands

• Can have object of different class than operator

• Must be a friend to access private or protected data

– Operator member functions of a specific class are called• Left operand of binary operator of same class

• Single operand of unitary operator of same class

Page 8: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

8

8.4 Operator Functions As Class Members Vs. As Friend Functions

• Commutative operators– May want + to be commutative

• So both “a + b” and “b + a” work

– Suppose we have two different classes

– Overloaded operator can only be member function when its class is on left• HugeIntClass + Long int• Can be member function

– When other way, need a non-member overload function• Long int + HugeIntClass

Page 9: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

9

8.5 Overloading Stream-Insertion and Stream-Extraction Operators

• << and >>– Already overloaded to process each built-in type

– Can also process a user-defined class

• Example program– Class PhoneNumber

• Holds a telephone number

– Print out formatted number automatically• (123) 456-7890

Page 10: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline10

fig08_03.cpp(1 of 3)

1 // Fig. 8.3: fig08_03.cpp2 // Overloading the stream-insertion and 3 // stream-extraction operators.4 #include <iostream>5 6 using std::cout;7 using std::cin;8 using std::endl;9 using std::ostream;10 using std::istream;11 12 #include <iomanip>13 14 using std::setw;15 16 // PhoneNumber class definition17 class PhoneNumber {18 friend ostream &operator<<( ostream&, const PhoneNumber & );19 friend istream &operator>>( istream&, PhoneNumber & ); 20 21 private:22 char areaCode[ 4 ]; // 3-digit area code and null23 char exchange[ 4 ]; // 3-digit exchange and null24 char line[ 5 ]; // 4-digit line and null25 26 }; // end class PhoneNumber

Notice function prototypes for overloaded operators >> and <<

They must be non-member friend functions, since the object of class Phonenumber appears on the right of the operator.

cin << objectcout >> object

Page 11: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline11

fig08_03.cpp(2 of 3)

27 28 // overloaded stream-insertion operator; cannot be 29 // a member function if we would like to invoke it with 30 // cout << somePhoneNumber; 31 ostream &operator<<( ostream &output, const PhoneNumber &num )32 { 33 output << "(" << num.areaCode << ") " 34 << num.exchange << "-" << num.line; 35 36 return output; // enables cout << a << b << c; 37 38 } // end function operator<< 39 40 // overloaded stream-extraction operator; cannot be 41 // a member function if we would like to invoke it with 42 // cin >> somePhoneNumber; 43 istream &operator>>( istream &input, PhoneNumber &num ) 44 { 45 input.ignore(); // skip ( 46 input >> setw( 4 ) >> num.areaCode; // input area code 47 input.ignore( 2 ); // skip ) and space48 input >> setw( 4 ) >> num.exchange; // input exchange 49 input.ignore(); // skip dash (-) 50 input >> setw( 5 ) >> num.line; // input line 51 52 return input; // enables cin >> a >> b >> c;

The expression:cout << phone; is interpreted as the function call:operator<<(cout, phone);

output is an alias for cout.

This allows objects to be cascaded.cout << phone1 << phone2;first calls operator<<(cout, phone1), and returns cout.

Next, cout << phone2 executes.

ignore() skips specified number of characters from input (1 by default).

Stream manipulator setw restricts number of characters read. setw(4) allows 3 characters to be read, leaving room for the null character.

Page 12: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline12

fig08_03.cpp(3 of 3)

fig08_03.cppoutput (1 of 1)

53 54 } // end function operator>> 55 56 int main()57 {58 PhoneNumber phone; // create object phone59 60 cout << "Enter phone number in the form (123) 456-7890:\n";61 62 // cin >> phone invokes operator>> by implicitly issuing63 // the non-member function call operator>>( cin, phone )64 cin >> phone; 65 66 cout << "The phone number entered was: " ;67 68 // cout << phone invokes operator<< by implicitly issuing69 // the non-member function call operator<<( cout, phone )70 cout << phone << endl; 71 72 return 0;73 74 } // end main

Enter phone number in the form (123) 456-7890:

(800) 555-1212

The phone number entered was: (800) 555-1212

Page 13: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

13

8.6 Overloading Unary Operators

• Overloading unary operators– Non-static member function, no arguments

– Non-member function, one argument• Argument must be class object or reference to class object

– Remember, static functions only access static data

Page 14: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

14

8.6 Overloading Unary Operators

• Upcoming example (8.10)– Overload ! to test for empty string

– If non-static member function, needs no arguments• !s becomes s.operator!()

class String {public: bool operator!() const; ...};

– If non-member function, needs one argument• s! becomes operator!(s) class String { friend bool operator!( const String & ) ...}

Page 15: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

15

8.7 Overloading Binary Operators

• Overloading binary operators– Non-static member function, one argument

– Non-member function, two arguments• One argument must be class object or reference

• Upcoming example– If non-static member function, needs one argument

class String {public: const String &operator+=( const String & ); ...};

– y += z equivalent to y.operator+=( z )

Page 16: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

16

8.7 Overloading Binary Operators

• Upcoming example– If non-member function, needs two arguments

– Example:class String { friend const String &operator+=( String &, const String & ); ...};

– y += z equivalent to operator+=( y, z )

Page 17: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

17

8.8 Converting between Types

• Casting– Traditionally, cast integers to floats, etc.

– May need to convert between user-defined types

• Cast operator (conversion operator)– Convert from

• One class to another

• Class to built-in type (int, char, etc.)

– Must be non-static member function• Cannot be friend

– Do not specify return type• Implicitly returns type to which you are converting

Page 18: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

18

8.8 Converting between Types

• Example– Prototype

A::operator char *() const;• Casts class A to a temporary char *• (char *)s calls s.operator char*()

– Also• A::operator int() const;• A::operator OtherClass() const;

Page 19: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

19

8.8 Converting between Types

• Casting can prevent need for overloading– Suppose class String can be cast to char *– cout << s; // s is a String

• Compiler implicitly converts s to char *• Do not have to overload <<

– Compiler can only do 1 cast

Page 20: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

20

8.9 Overloading ++ and --

• Increment/decrement operators can be overloaded– Add 1 to a Date object, d1– Prototype (member function)

• Date &operator++();• ++d1 same as d1.operator++()

– Prototype (non-member)• Friend Date &operator++( Date &);• ++d1 same as operator++( d1 )

Page 21: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

21

8.9 Overloading ++ and --

• To distinguish pre/post increment– Post increment has a dummy parameter

• int of 0

– Prototype (member function)• Date operator++( int );• d1++ same as d1.operator++( 0 )

– Prototype (non-member)• friend Date operator++( Data &, int );• d1++ same as operator++( d1, 0 )

– Integer parameter does not have a name• Not even in function definition

Page 22: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

22

8.9 Overloading ++ and --

• Return values– Preincrement

• Returns by reference (Date &)

• lvalue (can be assigned)

– Postincrement• Returns by value

• Returns temporary object with old value

• rvalue (cannot be on left side of assignment)

• Decrement operator analogous

Page 23: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc. All rights reserved.

23

8.10 Case Study: A Date Class

• Example Date class– Overloaded increment operator

• Change day, month and year

– Overloaded += operator

– Function to test for leap years

– Function to determine if day is last of month

Page 24: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline24

date1.h (1 of 2)

1 // Fig. 8.10: date1.h2 // Date class definition.3 #ifndef DATE1_H4 #define DATE1_H5 #include <iostream>6 7 using std::ostream;8 9 class Date {10 friend ostream &operator<<( ostream &, const Date & );11 12 public:13 Date( int m = 1, int d = 1, int y = 1900 ); // constructor14 void setDate( int, int, int ); // set the date15 16 Date &operator++(); // preincrement operator 17 Date operator++( int ); // postincrement operator18 19 const Date &operator+=( int ); // add days, modify object20 21 bool leapYear( int ) const; // is this a leap year?22 bool endOfMonth( int ) const; // is this end of month?

Note difference between pre and post increment.

Page 25: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline25

date1.h (2 of 2)

23 24 private:25 int month;26 int day;27 int year;28 29 static const int days[]; // array of days per month30 void helpIncrement(); // utility function31 32 }; // end class Date33 34 #endif

Page 26: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline26

date1.cpp (1 of 5)

1 // Fig. 8.11: date1.cpp2 // Date class member function definitions.3 #include <iostream>4 #include "date1.h"5 6 // initialize static member at file scope; 7 // one class-wide copy8 const int Date::days[] = 9 { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };10 11 // Date constructor12 Date::Date( int m, int d, int y ) 13 { 14 setDate( m, d, y ); 15 16 } // end Date constructor17 18 // set month, day and year19 void Date::setDate( int mm, int dd, int yy )20 {21 month = ( mm >= 1 && mm <= 12 ) ? mm : 1;22 year = ( yy >= 1900 && yy <= 2100 ) ? yy : 1900;23

Page 27: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline27

date1.cpp (2 of 5)

24 // test for a leap year25 if ( month == 2 && leapYear( year ) )26 day = ( dd >= 1 && dd <= 29 ) ? dd : 1;27 else28 day = ( dd >= 1 && dd <= days[ month ] ) ? dd : 1;29 30 } // end function setDate31 32 // overloaded preincrement operator 33 Date &Date::operator++() 34 { 35 helpIncrement(); 36 37 return *this; // reference return to create an lvalue38 39 } // end function operator++ 40 41 // overloaded postincrement operator; note that the dummy42 // integer parameter does not have a parameter name 43 Date Date::operator++( int ) 44 { 45 Date temp = *this; // hold current state of object 46 helpIncrement(); 47 48 // return unincremented, saved, temporary object 49 return temp; // value return; not a reference return50 51 } // end function operator++

Postincrement updates object and returns a copy of the original. Do not return a reference to temp, because it is a local variable that will be destroyed.

Also note that the integer parameter does not have a name.

Page 28: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline28

date1.cpp (3 of 5)

52 53 // add specified number of days to date54 const Date &Date::operator+=( int additionalDays )55 {56 for ( int i = 0; i < additionalDays; i++ )57 helpIncrement();58 59 return *this; // enables cascading60 61 } // end function operator+=62 63 // if the year is a leap year, return true; 64 // otherwise, return false65 bool Date::leapYear( int testYear ) const66 {67 if ( testYear % 400 == 0 || 68 ( testYear % 100 != 0 && testYear % 4 == 0 ) )69 return true; // a leap year70 else71 return false; // not a leap year72 73 } // end function leapYear74

Page 29: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline29

date1.cpp (4 of 5)

75 // determine whether the day is the last day of the month76 bool Date::endOfMonth( int testDay ) const77 {78 if ( month == 2 && leapYear( year ) )79 return testDay == 29; // last day of Feb. in leap year80 else81 return testDay == days[ month ];82 83 } // end function endOfMonth84 85 // function to help increment the date86 void Date::helpIncrement()87 {88 // day is not end of month89 if ( !endOfMonth( day ) )90 ++day;91 92 else 93 94 // day is end of month and month < 1295 if ( month < 12 ) {96 ++month;97 day = 1;98 }99

Page 30: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline30

date1.cpp (5 of 5)

100 // last day of year101 else {102 ++year;103 month = 1;104 day = 1;105 }106 107 } // end function helpIncrement108 109 // overloaded output operator110 ostream &operator<<( ostream &output, const Date &d )111 {112 static char *monthName[ 13 ] = { "", "January",113 "February", "March", "April", "May", "June",114 "July", "August", "September", "October",115 "November", "December" };116 117 output << monthName[ d.month ] << ' '118 << d.day << ", " << d.year;119 120 return output; // enables cascading121 122 } // end function operator<<

Page 31: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline31

fig08_12.cpp(1 of 2)

1 // Fig. 8.12: fig08_12.cpp2 // Date class test program.3 #include <iostream>4 5 using std::cout;6 using std::endl;7 8 #include "date1.h" // Date class definition9 10 int main()11 {12 Date d1; // defaults to January 1, 190013 Date d2( 12, 27, 1992 );14 Date d3( 0, 99, 8045 ); // invalid date15 16 cout << "d1 is " << d1 << "\nd2 is " << d217 << "\nd3 is " << d3;18 19 cout << "\n\nd2 += 7 is " << ( d2 += 7 );20 21 d3.setDate( 2, 28, 1992 );22 cout << "\n\n d3 is " << d3;23 cout << "\n++d3 is " << ++d3;24 25 Date d4( 7, 13, 2002 );

Page 32: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline32

fig08_12.cpp(2 of 2)

26 27 cout << "\n\nTesting the preincrement operator:\n"28 << " d4 is " << d4 << '\n'; 29 cout << "++d4 is " << ++d4 << '\n'; 30 cout << " d4 is " << d4; 31 32 cout << "\n\nTesting the postincrement operator:\n"33 << " d4 is " << d4 << '\n'; 34 cout << "d4++ is " << d4++ << '\n'; 35 cout << " d4 is " << d4 << endl; 36 37 return 0;38 39 } // end main

Page 33: 2003 Prentice Hall, Inc. All rights reserved. 1 Chapter 8 - Operator Overloading Outline 8.1 Introduction 8.2 Fundamentals of Operator Overloading 8.3.

2003 Prentice Hall, Inc.All rights reserved.

Outline33

fig08_12.cppoutput (1 of 1)

d1 is January 1, 1900

d2 is December 27, 1992

d3 is January 1, 1900

d2 += 7 is January 3, 1993

d3 is February 28, 1992

++d3 is February 29, 1992

Testing the preincrement operator:

d4 is July 13, 2002

++d4 is July 14, 2002

d4 is July 14, 2002

Testing the postincrement operator:

d4 is July 14, 2002

d4++ is July 14, 2002

d4 is July 15, 2002