1 / 20

Polymorphism

Polymorphism. ITK 169 Fall 2003. Many Forms. Polymorphism refers to the ability to associate multiple meanings to one function name. Within inherited classes, we can redefine functions or override functions (using virtual functions).

willis
Download Presentation

Polymorphism

An Image/Link below is provided (as is) to download presentation Download Policy: Content on the Website is provided to you AS IS for your information and personal use and may not be sold / licensed / shared on other websites without getting consent from its author. Content is provided to you AS IS for your information and personal use only. Download presentation by click this link. While downloading, if for some reason you are not able to download a presentation, the publisher may have deleted the file from their server. During download, if you can't get a presentation, the file might be deleted by the publisher.

E N D

Presentation Transcript


  1. Polymorphism ITK 169 Fall 2003

  2. Many Forms • Polymorphism refers to the ability to associate multiple meanings to one function name. • Within inherited classes, we can redefine functions or override functions (using virtual functions). • Virtual functions give us the ability to associate multiple meanings with one function name by means of the mechanism oflate binding.

  3. Virtual Funcitons • A virtual function is a function that, in a sense, may be used prior to definition. • Consider a base class Figure (with one data member center) and various derived classes: rectangle, oval, circle, etc. • Each figure class will add various additional data: • rectangle - width, length • oval - large width, a short width • circle - radius • Each of these classes needs a draw member function, with each implemented differently. • No new news here…

  4. The Figure class may have functions that apply to all figures. • Suppose Figure has a member function Figure::center that moves a figure to the center of the screen by two steps: • erasing the existing figure, • redrawing the figure at the center of the screen • The Figure::center member function uses the Figure::draw function to redraw the figure at the center. • But how can this happen? Would all figures be drawn in the same way?

  5. Virtual Functions • The inherited function center (unless something special is done) will use the definition of draw in class Figure, and that version of draw won't work for all figures. • When drawing a triangle, we want the inherited function draw to use Triangle::draw, not Figure::draw. • If the function draw is declared as a virtual function in class Figure, then things work correctly. How? • By declaring a member function virtual in a base class, we are telling the compiler to wait until this function is used in a program to decide what implementation to use. • This is called late binding or dynamic binding.

  6. Consider the following classes: class Pet { protected: string name; pubic: virtual void print( ); }; class Dog : public Pet { private: string breed; public: virtual void print( ); }; • And these variable declarations: Dog vdog; Pet vpet;

  7. Anything Dog object is also a Pet. Thus these assignments should be reasonable and legal. vdog.name = "Tiny"; vdog.breed = "Great Dane"; vpet = vdog; • This is legal, but there are problems. The breed field is lost. The variable vpet has type Pet, and only members of Pet are available to the object vpet, regardless of what was assigned to it. This is the "slicing problem". Here is an attempted access and an error message: cout << vpet.breed; //Error: class Pet has no member named breed.

  8. C++ provides a mechanism to treat a Dog as a Pet without slicing away the derived class attributes of breed and the member function(s) associated with class Dog. Consider the declarations: Pet *ppet; Dog *pdog; • Using pointers and dynamic variables we can treat Tiny as a Pet yet keep the breed: pdog = new Dog; pdog->name = "Tiny"; pdog->breed = "Great Dane"; ppet = pdog;

  9. We can still access the breed data member of the node pointed to by ppet, but not directly: • Suppose Dog::print( ); • is defined: void Dog::print( ) { cout << "name: " << name << endl; cout << "breed: " << breed << endl; } The statement ppet->print( ); will cause the following to be printed: name: Tiny breed: Great Dane by virtue of the fact that print( ) is a virtual member of Pet and Dog.

  10. Virtual Table • The function, print( ) was declared virtualin class Pet, so when the system sees the call ppet->print( ); it looks in the virtual table for classes Pet and Dog, sees that ppet points to an object of type Dog, and calls the function Dog::print( ) instead of the code for Pet::print( )

  11. OO programming with dynamic variables • These rules should help: 1)If p_parent is a pointer of a base class and p_child is a pointer of a derived class, then the following assignment of pointers is allowed: p_parent = p_child; Moreover, none of the data members or member functions of the dynamic variable being pointed to by p_child will be lost. 2)Although all the extra fields of the dynamic variable are there, you will need virtual member functions to access them.

  12. We have advocated developing incrementally. • This means code a little and test a little, then code a little more then test some more. • This is definitely not the case for virtual member functions. • An attempt to compile a program with a class that has even one virtual function that does not have an implementation, results in some very hard-to-understand error messages, even if you do not call the undefined member functions! • An error message that one compiler gives is "undefined reference to Class_Name virtual table."

  13. Solution • You should implement all virtual functions prior to compiling, even if it is just a dummy, as in class A { public: // constructors virtual void foo( ); // the rest of the class members }; void A::foo( ) { // body intentionally empty }

  14. Summary • Inheritance provides a tool for code reuse by deriving one class from another, adding features to the derived. • Derived class object inherit all the members of the base class, and may add members. • Late binding means that the decision of which version of a member function is appropriate is decided at runtime. Virtual functions are what C++ uses to achieve late binding. • A protected member in the base class is directly available to a publicly derived class's member functions.

  15. Program 6 • We will use virtual functions and late binding in program 6. • To accomplish this we will use a list of pointers to base (parent) class objects. • We will actually store pointers to child classes in this list. • When we access the pointers and use them to call functions, the specific function will be dependent on the actual pointer type (child class).

  16. Consider This Base Class class license { protected: string name; string address; public: license(); license(string nname, string naddress); virtual void printOne(ostream& out); };

  17. With This Child Class class driversL : public license { protected: int height; int weight; char vehicleType; public: driversL(); driversL( string nname, string naddress, int nheight, int nweight, char nvehType); virtual void printOne(ostream& out); };

  18. And This Child Class class petL : public license { protected: char animalType; string petName; bool shotsVerified; public: petL(); petL( string nname, string naddress, char anType, string pname, bool shots); virtual void printOne(ostream& out); };

  19. In main() We Have • An array of license pointers: larr. • In that array we can store either • license pointers, • driversL pointers, or • petL pointers. • What happens in this loop? for(int i = 0; i < size; i++) larr[i]->printOne(cout);

  20. Error Checking with new lpt = new license("mary", "normal"); if(lpt == NULL) { cout << "error allocating memory" << endl; exit(1); }

More Related