480 likes | 748 Views
Chapter 15: Polymorphism and Virtual Functions. Starting Out with C++ Early Objects Seventh Edition by Tony Gaddis, Judy Walters, and Godfrey Muganda. Animal. Cat. Dog. Poodle. 15.1 Type Compatibility in Inheritance Hierarchies. Classes in a program may be part of an
E N D
Chapter 15: Polymorphism and Virtual Functions Starting Out with C++ Early Objects Seventh Edition by Tony Gaddis, Judy Walters, and Godfrey Muganda
Animal Cat Dog Poodle 15.1 Type Compatibility in Inheritance Hierarchies • Classes in a program may be part of an inheritance hierarchy • Classes lower in the hierarchy are special cases of those above
Type Compatibility in Inheritance • A pointer to a derived class can be assigned to a pointer to a base class. Another way to say this is: • A base class pointer can point to derived class objects Animal *pA=new Cat;
Type Compatibility in Inheritance • Assigning a base class pointer to a derived class pointer requires a cast Animal *pA = new Cat; Cat *pC; pC = static_cast<Cat *>(pA); • The base class pointer must already point to a derived class object for this to work
Using Type Casts with Base Class Pointers C++ uses the declared type of a pointer to determine access to the members of the pointed-to object If an object of a derived class is pointed to by a base class pointer, all members of the derived class may not be accessible Type cast the base class pointer to the derived class (via static_cast) in order to access members that are specific to the derived class See project BasePointerCasting
15.2 Polymorphism and Virtual Member Functions Polymorphic code: Code that behaves differently when it acts on objects of different types Virtual Member Function: The C++ mechanism for achieving polymorphism
Animal Cat Dog Poodle Polymorphism Consider the Animal, Cat, Dog hierarchy where each class has its own version of the member function id( )
Polymorphism class Animal{ public:voidid(){cout<<"animal";} } class Cat : public Animal{ public:void id(){cout << "cat";} } class Dog : public Animal{ public:void id(){cout << "dog";} }
Polymorphism • Consider the collection of different Animal objects Animal *pA[] = {newAnimal,newDog, newCat}; and accompanying code for(int k=0; k<3; k++) pA[k]->id(); • Prints: animal animal animal, ignoring the more specific versions of id() in Dog and Cat • See project NonPolymorphic
Polymorphism Preceding code is not polymorphic: it behaves the same way even though Animal, Dog and Cat have different types and different id() member functions Polymorphic code would have printed "animaldogcat" instead of "animalanimalanimal"
Polymorphism • The code is not polymorphic because in the expression pA[k]->id() the compiler sees only the type of the pointer pA[k], which is pointer to Animal • Compiler does not see type of actual object pointed to, which may be Animal, or Dog, or Cat
Inheritance & Polymorphism Class Hierarchies • PassFailActivityinheritsscorefromGradedActivity and has its own member variable minPassingScore • PassFailExam inherits score and minPassingScore from PassFailActivity and has its own member variables numQuestions, pointsEach,numMissed • PassFailActivity redefines function getLetterGrade so that the the letter grade is either P or F and not A,B,C or D as computed by getLetterGrade in GradedActivity
Inheritance & Polymorphism GradedActivity .h and .cpp #include "GradedActivity.h" //*********************************************** //Member function GradedActivity::getLetterGrade* //*********************************************** char GradedActivity::getLetterGrade() const { char letterGrade; // To hold the letter grade if (score > 89) letterGrade = 'A'; else if (score > 79) letterGrade = 'B'; else if (score > 69) letterGrade = 'C'; else if (score > 59) letterGrade = 'D'; else letterGrade = 'F'; return letterGrade; } #ifndef GRADEDACTIVITY_H #define GRADEDACTIVITY_H // GradedActivity class declaration class GradedActivity { protected: double score; // To hold the numeric score public: // Default constructor GradedActivity() { score = 0.0; } // Constructor GradedActivity(double s) { score = s; } // Mutator function void setScore(double s) { score = s; } // Accessor functions double getScore() const { return score; } char getLetterGrade() const; }; #endif
Inheritance & Polymorphism FinalExam .h and .cpp #include "FinalExam.h" //******************************************************** // set function * // The parameters are the number of questions and the * // number of questions missed. * //******************************************************** void FinalExam::set(int questions, int missed) { double numericScore; // To hold the numeric score // Set the number of questions and number missed. numQuestions = questions; numMissed = missed; // Calculate the points for each question. pointsEach = 100.0 / numQuestions; // Calculate the numeric score for this exam. numericScore = 100.0 - (missed * pointsEach); // Call the inherited setScore function to set // the numeric score. setScore(numericScore); } #ifndef FINALEXAM_H #define FINALEXAM_H #include "GradedActivity.h" class FinalExam : public GradedActivity { private: int numQuestions; // Number of questions double pointsEach; // Points for each question int numMissed; // Number of questions missed public: FinalExam() { numQuestions = 0; pointsEach = 0.0; numMissed = 0; } FinalExam(int questions, int missed) { set(questions, missed); } void set(int, int); // Defined in FinalExam.cpp double getNumQuestions() const { return numQuestions; } double getPointsEach() const { return pointsEach; } int getNumMissed() const { return numMissed; } }; #endif
Inheritance & Polymorphism PassFailActivity .h and .cpp #include "PassFailActivity.h" //****************************************************** // Member function PassFailActivity::getLetterGrade * // This function returns 'P' if the score is passing, * // otherwise it returns 'F'. * //****************************************************** char PassFailActivity::getLetterGrade() const { char letterGrade; if (score >= minPassingScore) letterGrade = 'P'; else letterGrade = 'F'; return letterGrade; } #ifndef PASSFAILACTIVITY_H #define PASSFAILACTIVITY_H #include "GradedActivity.h" class PassFailActivity : public GradedActivity { protected: double minPassingScore; // Minimum passing score. public: // Default constructor PassFailActivity() : GradedActivity() { minPassingScore = 0.0; } // Constructor PassFailActivity(double mps) : GradedActivity() { minPassingScore = mps; } // Mutator void setMinPassingScore(double mps) { minPassingScore = mps; } // Accessors double getMinPassingScore() const { return minPassingScore; } char getLetterGrade() const; }; #endif
Inheritance & Polymorphism PassFailExam .h and .cpp #include "PassFailExam.h" //******************************************************** // set function * // The parameters are the number of questions and the * // number of questions missed. * //******************************************************** void PassFailExam::set(int questions, int missed) { double numericScore; // To hold the numeric score // Set the number of questions and number missed. numQuestions = questions; numMissed = missed; // Calculate the points for each question. pointsEach = 100.0 / numQuestions; // Calculate the numeric score for this exam. numericScore = 100.0 - (missed * pointsEach); // Call the inherited setScore function to set // the numeric score. setScore(numericScore); } #ifndef PASSFAILEXAM_H #define PASSFAILEXAM_H #include "PassFailActivity.h" class PassFailExam : public PassFailActivity { private: int numQuestions; // Number of questions double pointsEach; // Points for each question int numMissed; // Number of questions missed public: // Default constructor PassFailExam() : PassFailActivity() { numQuestions = 0; pointsEach = 0.0; numMissed = 0; } // Constructor PassFailExam(int questions, int missed, double mps) : PassFailActivity(mps) { set(questions, missed); } // Mutator function void set(int, int); // Defined in PassFailExam.cpp // Accessor functions double getNumQuestions() const { return numQuestions; } double getPointsEach() const { return pointsEach; } int getNumMissed() const { return numMissed; } }; #endif
Inheritance & Polymorphism Consider this function Because the parameter in the displayGrade function is a GradedActivity reference variable, it can reference any object that is derived from GradedActivity. That means we can pass a GradedActivity object, a FinalExam object, a PassFailExam object, or any other object that is derived from GradedActivity. A problem occurs in this program however...
Inheritance & Polymorphism • As you can see from the example output, the getLetterGrade member function returned ‘C’ instead of ‘P’. • This is because the GradedActivity class’ getLetterGrade function was executed instead of the PassFailActivity class’s version of the function.
Inheritance & Polymorphism Static Binding • Program 15-9 displays 'C' instead of 'P' because the call to the getLetterGrade function is statically bound (at compile time) with the GradedActivity class's version of the function. • Thus, the actual letter grade is computed instead of the P/F grade • We can remedy this by making the getLetterGrade function virtual in GradedActivity.
Virtual Functions Declaring a function virtual will make the compiler check the type of each object to see if it defines a more specific version of the virtual function At runtime, C++ determines the type of object making the call, and binds the function to the appropriate version of the function.
Virtual Functions If the member functions id()are declared virtual, then the code Animal *pA[] = {newAnimal, newDog,newCat}; for(int k=0; k<3; k++) pA[k]->id(); will print animaldogcat
Virtual Functions How to declare a member function virtual: class Animal{ public:virtualvoidid(){cout<<"animal";} } class Cat : public Animal{ public:virtualvoid id(){cout << "cat";} } class Dog : public Animal{ public:virtualvoid id(){cout << "dog";} }
Function Binding In pA[k]->id(), Compiler must choose which version of id() to use: There are different versions in the Animal, Dog, and Cat classes Function binding is the process of determining which function definition to use for a particular function call The alternatives are static and dynamic binding
Static Binding Static binding chooses the function in the class of the base class pointer, ignoring any versions in the class of the object actually pointed to Static binding is done at compile time
Dynamic Binding Dynamic Binding determines the function to be invoked at execution time Can look at the actual class of the object pointed to and choose the most specific version of the function Dynamic binding is used to bind virtual functions As long as the array or pointer that holds the objects of the child classes has a virtual function, dynamic binding will occur even if the child functions are not virtual See project Polymorphic-Virtual
Inheritance & Polymorphism Updated Version of GradedActivity The function is now virtual. The function also becomes virtual in all derived classes automatically!
Inheritance & Polymorphism If we recompile our program with the updated versions of the classes, we will get the right output, shown here (project GradedActivity-3): This type of behavior is known as polymorphism. The term polymorphism means the ability to take many forms. The program demonstrates polymorphism by passing objects of the GradedActivity and PassFailExam classes to the displayGrade function.
Inheritance & Polymorphism Virtual Destructors • It's a good idea to make destructors virtual if the class could ever become a base class. • Otherwise, the compiler will perform static binding on the destructor if the class ever is derived from. • If the derived class is pointed to by an object of the base class, only the base class destructor will be called • Making the base class destructor virtual will enable both destructors to execute • When a base class function is declared virtual, all overridden versions of the function in derived classes automatically become virtual • Including a virtual destructor in a base class, even one that does nothing, will ensure that any derived class destructors will also be virtual • See project VirtualDestructor
15.3 Abstract Base Classes and Pure Virtual Functions An abstract class is a class that contains no objects that are not members of subclasses (derived classes) For example, in real life, Animal is an abstract class: there are no animals that are not dogs, or cats, or lions…
Abstract Base Classes and Pure Virtual Functions Abstract classes are an organizational tool: useful in organizing inheritance hierarchies Abstract classes can be used to specify an interface that must be implemented by all subclasses
Abstract Functions The member functions specified in an abstract class do not have to be implemented The implementation is left to the subclasses In C++, an abstract class is a class with at least one abstract member function
Pure Virtual Functions • In C++, a member function of a class is declared to be an abstract function by making it virtual and replacing its body with = 0; class Animal{ public: virtual void id()=0; }; • A virtual function with its body omitted and replaced with =0 is called a pure virtualfunction, or an abstract function
Abstract Classes An abstract class can not be instantiated An abstract class can only be inherited from: that is, you can derive classes from it Classes derived from abstract classes must override the pure virtual function with a concrete member function before they can be instantiated. See project AbstractBaseClass2
Inheritance & Polymorphism Abstract Base Classes and Pure Virtual Functions • In Student.h we have a pure virtual function virtual int getRemainingHours() const = 0; • This is defined in CsStudent.cpp int CsStudent::getRemainingHours() const { int reqHours, // Total required hours remainingHours; // Remaining hours // Calculate the required hours. reqHours = MATH_HOURS + CS_HOURS + GEN_ED_HOURS; // Calculate the remaining hours. remainingHours = reqHours - (mathHours + csHours + genEdHours); // Return the remaining hours. return remainingHours; }
Inheritance & Polymorphism Abstract Base Classes and Pure Virtual Functions • There is no meaning to calculating the remaining hours for a general student – each major has its own requirements • Thus, the pure virtual function in class Student which is given specific definition each derived class • See project AbstractBaseClass
Inheritance & Polymorphism class square class rectSolid class cube Multiple Inheritance • A derived class can have more than one base class • Each base class can have its own access specification in derived class's definition: class cube : public square, public rectSolid;
Inheritance & Polymorphism Multiple Inheritance • Arguments can be passed to both base classes' constructors: cube::cube(int side) : square(side), rectSolid(side, side, side); DateTime::DateTime(int dy, int mon, int yr, int hr, int mt, int sc) : Date(dy, mon, yr), Time(hr, mt, sc) • The order the base class constructor calls appear in the list does not matter. They are called in order of inheritance. • Base class constructors are called in order they are listed in the first line of the class declaration
Inheritance & Polymorphism Multiple Inheritance • Problem: what if base classes have member variables/functions with the same name? • Solutions: • Derived class redefines the multiply-defined function • Derived class invokes member function in a particular base class using scope resolution operator :: • Compiler errors occur if derived class uses base class function without one of these solutions • See project MultipleInheritance
Chapter 15: Polymorphism and Virtual Functions Starting Out with C++ Early Objects Seventh Edition by Tony Gaddis, Judy Walters, and Godfrey Muganda