Presentation is loading. Please wait.

Presentation is loading. Please wait.

Outline 1 Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü.

Similar presentations


Presentation on theme: "Outline 1 Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü."— Presentation transcript:

1 Outline 1 Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü C ++ LECTURE 9 OOP: Polymorphism

2 2 Object-Oriented Programming: Polymorphism Outline 10.1 Introduction 10.2 Relationships Among Objects in an Inheritance Hierarchy 10.2.1 Invoking Base-Class Functions from Derived-Class Objects 10.2.2 Aiming Derived-Class Pointers at Base-Class Objects 10.2.3 Derived-Class Member-Function Calls via Base-Class Pointers 10.2.4 Virtual Functions 10.3 Polymorphism Examples 10.4 Type Fields and switch Structures 10.5 Abstract Classes 10.6 Case Study: Inheriting Interface and Implementation

3 3 10.1 Introduction Polymorphism –“Program in the general” –Treat objects in same class hierarchy as if all base class –Virtual functions and dynamic binding Will explain how polymorphism works –Makes programs extensible New classes added easily, can still be processed In our examples –Use abstract base class Shape Defines common interface (functionality) Point, Circle and Cylinder inherit from Shape –Class Employee for a natural example

4 4 10.2 Relationships Among Objects in an Inheritance Hierarchy Previously (Section 9.4), –Circle inherited from Point –Manipulated Point and Circle objects using member functions Now –Invoke functions using base-class/derived-class pointers –Introduce virtual functions Key concept –Derived-class object can be treated as base-class object “is-a” relationship Base class is not a derived class object

5 5 10.2.1 Invoking Base-Class Functions from Derived-Class Objects Aim pointers (base, derived) at objects (base, derived) –Base pointer aimed at base object –Derived pointer aimed at derived object Both straightforward –Base pointer aimed at derived object “is a” relationship –Circle “is a” Point Will invoke base class functions –Function call depends on the class of the pointer/handle Does not depend on object to which it points With virtual functions, this can be changed (more later)

6 Outline 6 point.h (1 of 1) 1 // Fig. 10.1: point.h 2 // Point class definition represents an x-y coordinate pair. 3 #ifndef POINT_H 4 #define POINT_H 5 6 class Point { 7 8 public: 9 Point( int = 0, int = 0 ); // default constructor 10 11 void setX( int ); // set x in coordinate pair 12 int getX() const; // return x from coordinate pair 13 14 void setY( int ); // set y in coordinate pair 15 int getY() const; // return y from coordinate pair 16 17 void print() const; // output Point object 18 19 private: 20 int x; // x part of coordinate pair 21 int y; // y part of coordinate pair 22 23 }; // end class Point 24 25 #endif Base class print function.

7 Outline 7 point.cpp (1 of 2) 1 // Fig. 10.2: point.cpp 2 // Point class member-function definitions. 3 #include 4 5 using std::cout; 6 7 #include "point.h" // Point class definition 8 9 // default constructor 10 Point::Point( int xValue, int yValue ) 11 : x( xValue ), y( yValue ) 12 { 13 // empty body 14 15 } // end Point constructor 16 17 // set x in coordinate pair 18 void Point::setX( int xValue ) 19 { 20 x = xValue; // no need for validation 21 22 } // end function setX 23

8 Outline 8 point.cpp (2 of 2) 24 // return x from coordinate pair 25 int Point::getX() const 26 { 27 return x; 28 29 } // end function getX 30 31 // set y in coordinate pair 32 void Point::setY( int yValue ) 33 { 34 y = yValue; // no need for validation 35 36 } // end function setY 37 38 // return y from coordinate pair 39 int Point::getY() const 40 { 41 return y; 42 43 } // end function getY 44 45 // output Point object 46 void Point::print() const 47 { 48 cout << '[' << getX() << ", " << getY() << ']'; 49 50 } // end function print Output the x,y coordinates of the Point.

9 Outline 9 circle.h (1 of 1) 1 // Fig. 10.3: circle.h 2 // Circle class contains x-y coordinate pair and radius. 3 #ifndef CIRCLE_H 4 #define CIRCLE_H 5 6 #include "point.h" // Point class definition 7 8 class Circle : public Point { 9 10 public: 11 12 // default constructor 13 Circle( int = 0, int = 0, double = 0.0 ); 14 15 void setRadius( double ); // set radius 16 double getRadius() const; // return radius 17 18 double getDiameter() const; // return diameter 19 double getCircumference() const; // return circumference 20 double getArea() const; // return area 21 22 void print() const; // output Circle object 23 24 private: 25 double radius; // Circle's radius 26 27 }; // end class Circle 28 29 #endif Circle inherits from Point, but redefines its print function.

10 Outline 10 circle.cpp (1 of 3) 1 // Fig. 10.4: circle.cpp 2 // Circle class member-function definitions. 3 #include 4 5 using std::cout; 6 7 #include "circle.h" // Circle class definition 8 9 // default constructor 10 Circle::Circle( int xValue, int yValue, double radiusValue ) 11 : Point( xValue, yValue ) // call base-class constructor 12 { 13 setRadius( radiusValue ); 14 15 } // end Circle constructor 16 17 // set radius 18 void Circle::setRadius( double radiusValue ) 19 { 20 radius = ( radiusValue < 0.0 ? 0.0 : radiusValue ); 21 22 } // end function setRadius 23

11 Outline 11 circle.cpp (2 of 3) 24 // return radius 25 double Circle::getRadius() const 26 { 27 return radius; 28 29 } // end function getRadius 30 31 // calculate and return diameter 32 double Circle::getDiameter() const 33 { 34 return 2 * getRadius(); 35 36 } // end function getDiameter 37 38 // calculate and return circumference 39 double Circle::getCircumference() const 40 { 41 return 3.14159 * getDiameter(); 42 43 } // end function getCircumference 44 45 // calculate and return area 46 double Circle::getArea() const 47 { 48 return 3.14159 * getRadius() * getRadius(); 49 50 } // end function getArea

12 Outline 12 circle.cpp (3 of 3) 51 52 // output Circle object 53 void Circle::print() const 54 { 55 cout << "center = "; 56 Point::print(); // invoke Point's print function 57 cout << "; radius = " << getRadius(); 58 59 } // end function print Circle redefines its print function. It calls Point ’s print function to output the x,y coordinates of the center, then prints the radius.

13 Outline 13 fig10_05.cpp (1 of 3) 1 // Fig. 10.5: fig10_05.cpp 2 // Aiming base-class and derived-class pointers at base-class 3 // and derived-class objects, respectively. 4 #include 5 6 using std::cout; 7 using std::endl; 8 using std::fixed; 9 10 #include 11 12 using std::setprecision; 13 14 #include "point.h" // Point class definition 15 #include "circle.h" // Circle class definition 16 17 int main() 18 { 19 Point point( 30, 50 ); 20 Point *pointPtr = 0; // base-class pointer 21 22 Circle circle( 120, 89, 2.7 ); 23 Circle *circlePtr = 0; // derived-class pointer 24

14 Outline 14 fig10_05.cpp (2 of 3) 25 // set floating-point numeric formatting 26 cout << fixed << setprecision( 2 ); 27 28 // output objects point and circle 29 cout << "Print point and circle objects:" 30 << "\nPoint: "; 31 point.print(); // invokes Point's print 32 cout << "\nCircle: "; 33 circle.print(); // invokes Circle's print 34 35 // aim base-class pointer at base-class object and print 36 pointPtr = &point; 37 cout << "\n\nCalling print with base-class pointer to " 38 << "\nbase-class object invokes base-class print " 39 << "function:\n"; 40 pointPtr->print(); // invokes Point's print 41 42 // aim derived-class pointer at derived-class object 43 // and print 44 circlePtr = &circle; 45 cout << "\n\nCalling print with derived-class pointer to " 46 << "\nderived-class object invokes derived-class " 47 << "print function:\n"; 48 circlePtr->print(); // invokes Circle's print 49 Use objects and pointers to call the print function. The pointers and objects are of the same class, so the proper print function is called.

15 Outline 15 fig10_05.cpp (3 of 3) 50 // aim base-class pointer at derived-class object and print 51 pointPtr = &circle; 52 cout << "\n\nCalling print with base-class pointer to " 53 << "derived-class object\ninvokes base-class print " 54 << "function on that derived-class object:\n"; 55 pointPtr->print(); // invokes Point's print 56 cout << endl; 57 58 return 0; 59 60 } // end main Aiming a base-class pointer at a derived object is allowed (the Circle “is a” Point ). However, it calls Point ’s print function, determined by the pointer type. virtual functions allow us to change this.

16 Outline 16 fig10_05.cpp output (1 of 1) Print point and circle objects: Point: [30, 50] Circle: center = [120, 89]; radius = 2.70 Calling print with base-class pointer to base-class object invokes base-class print function: [30, 50] Calling print with derived-class pointer to derived-class object invokes derived-class print function: center = [120, 89]; radius = 2.70 Calling print with base-class pointer to derived-class object invokes base-class print function on that derived-class object: [120, 89]

17 17 10.2.2 Aiming Derived-Class Pointers at Base-Class Objects Previous example –Aimed base-class pointer at derived object Circle “is a” Point Aim a derived-class pointer at a base-class object –Compiler error No “is a” relationship Point is not a Circle Circle has data/functions that Point does not –setRadius (defined in Circle ) not defined in Point –Can cast base-object’s address to derived-class pointer Called downcasting (more in 10.9) Allows derived-class functionality

18 Outline 18 fig10_06.cpp (1 of 1) fig10_06.cpp output (1 of 1) 1 // Fig. 10.6: fig10_06.cpp 2 // Aiming a derived-class pointer at a base-class object. 3 #include "point.h" // Point class definition 4 #include "circle.h" // Circle class definition 5 6 int main() 7 { 8 Point point( 30, 50 ); 9 Circle *circlePtr = 0; 10 11 // aim derived-class pointer at base-class object 12 circlePtr = &point; // Error: a Point is not a Circle 13 14 return 0; 15 16 } // end main C:\cpphtp4\examples\ch10\fig10_06\Fig10_06.cpp(12) : error C2440: '=' : cannot convert from 'class Point *' to 'class Circle *' Types pointed to are unrelated; conversion requires reinterpret_cast, C-style cast or function-style cast

19 19 10.2.3 Derived-Class Member-Function Calls via Base-Class Pointers Handle (pointer/reference) –Base-pointer can aim at derived-object But can only call base-class functions –Calling derived-class functions is a compiler error Functions not defined in base-class Common theme –Data type of pointer/reference determines functions it can call

20 Outline 20 fig10_07.cpp (1 of 2) 1 // Fig. 10.7: fig10_07.cpp 2 // Attempting to invoke derived-class-only member functions 3 // through a base-class pointer. 4 #include "point.h" // Point class definition 5 #include "circle.h" // Circle class definition 6 7 int main() 8 { 9 Point *pointPtr = 0; 10 Circle circle( 120, 89, 2.7 ); 11 12 // aim base-class pointer at derived-class object 13 pointPtr = &circle; 14 15 // invoke base-class member functions on derived-class 16 // object through base-class pointer 17 int x = pointPtr->getX(); 18 int y = pointPtr->getY(); 19 pointPtr->setX( 10 ); 20 pointPtr->setY( 10 ); 21 pointPtr->print(); 22

21 Outline 21 fig10_07.cpp (2 of 2) 23 // attempt to invoke derived-class-only member functions 24 // on derived-class object through base-class pointer 25 double radius = pointPtr->getRadius(); 26 pointPtr->setRadius( 33.33 ); 27 double diameter = pointPtr->getDiameter(); 28 double circumference = pointPtr->getCircumference(); 29 double area = pointPtr->getArea(); 30 31 return 0; 32 33 } // end main These functions are only defined in Circle. However, pointPtr is of class Point.

22 Outline 22 fig10_07.cpp output (1 of 1) C:\cpphtp4\examples\ch10\fig10_07\fig10_07.cpp(25) : error C2039: 'getRadius' : is not a member of 'Point' C:\cpphtp4\examples\ch10\fig10_07\point.h(6) : see declaration of 'Point' C:\cpphtp4\examples\ch10\fig10_07\fig10_07.cpp(26) : error C2039: 'setRadius' : is not a member of 'Point' C:\cpphtp4\examples\ch10\fig10_07\point.h(6) : see declaration of 'Point' C:\cpphtp4\examples\ch10\fig10_07\fig10_07.cpp(27) : error C2039: 'getDiameter' : is not a member of 'Point' C:\cpphtp4\examples\ch10\fig10_07\point.h(6) : see declaration of 'Point' C:\cpphtp4\examples\ch10\fig10_07\fig10_07.cpp(28) : error C2039: 'getCircumference' : is not a member of 'Point' C:\cpphtp4\examples\ch10\fig10_07\point.h(6) : see declaration of 'Point' C:\cpphtp4\examples\ch10\fig10_07\fig10_07.cpp(29) : error C2039: 'getArea' : is not a member of 'Point' C:\cpphtp4\examples\ch10\fig10_07\point.h(6) : see declaration of 'Point'

23 23 10.2.4 Virtual Functions Typically, pointer-class determines functions virtual functions –Object (not pointer) determines function called Why useful? –Suppose Circle, Triangle, Rectangle derived from Shape Each has own draw function –To draw any shape Have base class Shape pointer, call draw Program determines proper draw function at run time (dynamically) Treat all shapes generically

24 24 10.2.4 Virtual Functions Declare draw as virtual in base class –Override draw in each derived class Like redefining, but new function must have same signature –If function declared virtual, can only be overridden virtual void draw() const; Once declared virtual, virtual in all derived classes –Good practice to explicitly declare virtual Dynamic binding –Choose proper function to call at run time –Only occurs off pointer handles If function called from object, uses that object’s definition

25 25 10.2.4 Virtual Functions Example –Redo Point, Circle example with virtual functions –Base-class pointer to derived-class object Will call derived-class function

26 Outline 26 point.h (1 of 1) 1 // Fig. 10.8: point.h 2 // Point class definition represents an x-y coordinate pair. 3 #ifndef POINT_H 4 #define POINT_H 5 6 class Point { 7 8 public: 9 Point( int = 0, int = 0 ); // default constructor 10 11 void setX( int ); // set x in coordinate pair 12 int getX() const; // return x from coordinate pair 13 14 void setY( int ); // set y in coordinate pair 15 int getY() const; // return y from coordinate pair 16 17 virtual void print() const; // output Point object 18 19 private: 20 int x; // x part of coordinate pair 21 int y; // y part of coordinate pair 22 23 }; // end class Point 24 25 #endif Print declared virtual. It will be virtual in all derived classes.

27 Outline 27 circle.h (1 of 1) 1 // Fig. 10.9: circle.h 2 // Circle class contains x-y coordinate pair and radius. 3 #ifndef CIRCLE_H 4 #define CIRCLE_H 5 6 #include "point.h" // Point class definition 7 8 class Circle : public Point { 9 10 public: 11 12 // default constructor 13 Circle( int = 0, int = 0, double = 0.0 ); 14 15 void setRadius( double ); // set radius 16 double getRadius() const; // return radius 17 18 double getDiameter() const; // return diameter 19 double getCircumference() const; // return circumference 20 double getArea() const; // return area 21 22 virtual void print() const; // output Circle object 23 24 private: 25 double radius; // Circle's radius 26 27 }; // end class Circle 28 29 #endif

28 Outline 28 fig10_10.cpp (1 of 3) 1 // Fig. 10.10: fig10_10.cpp 2 // Introducing polymorphism, virtual functions and dynamic 3 // binding. 4 #include 5 6 using std::cout; 7 using std::endl; 8 using std::fixed; 9 10 #include 11 12 using std::setprecision; 13 14 #include "point.h" // Point class definition 15 #include "circle.h" // Circle class definition 16 17 int main() 18 { 19 Point point( 30, 50 ); 20 Point *pointPtr = 0; 21 22 Circle circle( 120, 89, 2.7 ); 23 Circle *circlePtr = 0; 24

29 Outline 29 fig10_10.cpp (2 of 3) 25 // set floating-point numeric formatting 26 cout << fixed << setprecision( 2 ); 27 28 // output objects point and circle using static binding 29 cout << "Invoking print function on point and circle " 30 << "\nobjects with static binding " 31 << "\n\nPoint: "; 32 point.print(); // static binding 33 cout << "\nCircle: "; 34 circle.print(); // static binding 35 36 // output objects point and circle using dynamic binding 37 cout << "\n\nInvoking print function on point and circle " 38 << "\nobjects with dynamic binding"; 39 40 // aim base-class pointer at base-class object and print 41 pointPtr = &point; 42 cout << "\n\nCalling virtual function print with base-class" 43 << "\npointer to base-class object" 44 << "\ninvokes base-class print function:\n"; 45 pointPtr->print(); 46

30 Outline 30 fig10_10.cpp (3 of 3) 47 // aim derived-class pointer at derived-class 48 // object and print 49 circlePtr = &circle; 50 cout << "\n\nCalling virtual function print with " 51 << "\nderived-class pointer to derived-class object " 52 << "\ninvokes derived-class print function:\n"; 53 circlePtr->print(); 54 55 // aim base-class pointer at derived-class object and print 56 pointPtr = &circle; 57 cout << "\n\nCalling virtual function print with base-class" 58 << "\npointer to derived-class object " 59 << "\ninvokes derived-class print function:\n"; 60 pointPtr->print(); // polymorphism: invokes circle's print 61 cout << endl; 62 63 return 0; 64 65 } // end main At run time, the program determines that pointPtr is aiming at a Circle object, and calls Circle ’s print function. This is an example of polymorphism.

31 Outline 31 fig10_10.cpp output (1 of 1) Invoking print function on point and circle objects with static binding Point: [30, 50] Circle: Center = [120, 89]; Radius = 2.70 Invoking print function on point and circle objects with dynamic binding Calling virtual function print with base-class pointer to base-class object invokes base-class print function: [30, 50] Calling virtual function print with derived-class pointer to derived-class object invokes derived-class print function: Center = [120, 89]; Radius = 2.70 Calling virtual function print with base-class pointer to derived-class object invokes derived-class print function: Center = [120, 89]; Radius = 2.70

32 32 10.2.4 Virtual Functions Polymorphism –Same message, “print”, given to many objects All through a base pointer –Message takes on “many forms” Summary –Base-pointer to base-object, derived-pointer to derived Straightforward –Base-pointer to derived object Can only call base-class functions –Derived-pointer to base-object Compiler error Allowed if explicit cast made (more in section 10.9)

33 33 10.3 Polymorphism Examples Examples –Suppose Rectangle derives from Quadrilateral Rectangle more specific Quadrilateral Any operation on Quadrilateral can be done on Rectangle (i.e., perimeter, area) Suppose designing video game –Base class SpaceObject Derived Martian, SpaceShip, LaserBeam Base function draw –To refresh screen Screen manager has vector of base-class pointers to objects Send draw message to each object Same message has “many forms” of results

34 34 10.3 Polymorphism Examples Video game example, continued –Easy to add class Mercurian Inherits from SpaceObject Provides own definition for draw –Screen manager does not need to change code Calls draw regardless of object’s type Mercurian objects “plug right in” Polymorphism –Command many objects without knowing type –Extensible programs Add classes easily

35 35 10.4 Type Fields and switch Structures One way to determine object's class –Give base class an attribute shapeType in class Shape –Use switch to call proper print function Many problems –May forget to test for case in switch –If add/remove a class, must update switch structures Time consuming and error prone Better to use polymorphism –Less branching logic, simpler programs, less debugging

36 36 10.5 Abstract Classes Abstract classes –Sole purpose: to be a base class (called abstract base classes) –Incomplete Derived classes fill in "missing pieces" –Cannot make objects from abstract class However, can have pointers and references Concrete classes –Can instantiate objects –Implement all functions they define –Provide specifics

37 37 10.5 Abstract Classes Abstract classes not required, but helpful To make a class abstract –Need one or more "pure" virtual functions Declare function with initializer of 0 virtual void draw() const = 0; –Regular virtual functions Have implementations, overriding is optional –Pure virtual functions No implementation, must be overridden –Abstract classes can have data and concrete functions Required to have one or more pure virtual functions

38 38 10.5 Abstract Classes Abstract base class pointers –Useful for polymorphism Application example –Abstract class Shape Defines draw as pure virtual function –Circle, Triangle, Rectangle derived from Shape Each must implement draw –Screen manager knows that each object can draw itself Iterators (more Chapter 21) –Walk through elements in vector /array –Use base-class pointer to send draw message to each

39 39 10.6 Case Study: Inheriting Interface and Implementation Make abstract base class Shape –Pure virtual functions (must be implemented) getName, print Default implementation does not make sense –Virtual functions (may be redefined) getArea, getVolume –Initially return 0.0 If not redefined, uses base class definition –Derive classes Point, Circle, Cylinder

40 40 10.6 Case Study: Inheriting Interface and Implementation 0.0 = 0 0.0 "Point"[x,y] pr2pr2 0.0"Circle" center=[x,y]; radius=r 2 p r 2 +2 p rh pr2hpr2h "Cylinder" center=[x,y]; radius=r; height=h getAreaprintgetNamegetVolume Shape Point Circle Cylinder

41 Outline 41 shape.h (1 of 1) 1 // Fig. 10.12: shape.h 2 // Shape abstract-base-class definition. 3 #ifndef SHAPE_H 4 #define SHAPE_H 5 6 #include // C++ standard string class 7 8 using std::string; 9 10 class Shape { 11 12 public: 13 14 // virtual function that returns shape area 15 virtual double getArea() const; 16 17 // virtual function that returns shape volume 18 virtual double getVolume() const; 19 20 // pure virtual functions; overridden in derived classes 21 virtual string getName() const = 0; // return shape name 22 virtual void print() const = 0; // output shape 23 24 }; // end class Shape 25 26 #endif Virtual and pure virtual functions.

42 Outline 42 shape.cpp (1 of 1) 1 // Fig. 10.13: shape.cpp 2 // Shape class member-function definitions. 3 #include 4 5 using std::cout; 6 7 #include "shape.h" // Shape class definition 8 9 // return area of shape; 0.0 by default 10 double getArea() const 11 { 12 return 0.0; 13 14 } // end function getArea 15 16 // return volume of shape; 0.0 by default 17 double getVolume() const 18 { 19 return 0.0; 20 21 } // end function getVolume

43 Outline 43 point.h (1 of 2) 1 // Fig. 10.14: point.h 2 // Point class definition represents an x-y coordinate pair. 3 #ifndef POINT_H 4 #define POINT_H 5 6 #include "shape.h" // Shape class definition 7 8 class Point : public Shape { 9 10 public: 11 Point( int = 0, int = 0 ); // default constructor 12 13 void setX( int ); // set x in coordinate pair 14 int getX() const; // return x from coordinate pair 15 16 void setY( int ); // set y in coordinate pair 17 int getY() const; // return y from coordinate pair 18 19 // return name of shape (i.e., "Point" ) 20 virtual string getName() const; 21 22 virtual void print() const; // output Point object 23 Point only redefines getName and print, since getArea and getVolume are zero (it can use the default implementation).

44 Outline 44 point.h (2 of 2) 24 private: 25 int x; // x part of coordinate pair 26 int y; // y part of coordinate pair 27 28 }; // end class Point 29 30 #endif

45 Outline 45 point.cpp (1 of 3) 1 // Fig. 10.15: point.cpp 2 // Point class member-function definitions. 3 #include 4 5 using std::cout; 6 7 #include "point.h" // Point class definition 8 9 // default constructor 10 Point::Point( int xValue, int yValue ) 11 : x( xValue ), y( yValue ) 12 { 13 // empty body 14 15 } // end Point constructor 16 17 // set x in coordinate pair 18 void Point::setX( int xValue ) 19 { 20 x = xValue; // no need for validation 21 22 } // end function setX 23

46 Outline 46 point.cpp (2 of 3) 24 // return x from coordinate pair 25 int Point::getX() const 26 { 27 return x; 28 29 } // end function getX 30 31 // set y in coordinate pair 32 void Point::setY( int yValue ) 33 { 34 y = yValue; // no need for validation 35 36 } // end function setY 37 38 // return y from coordinate pair 39 int Point::getY() const 40 { 41 return y; 42 43 } // end function getY 44

47 Outline 47 point.cpp (3 of 3) 45 // override pure virtual function getName: return name of Point 46 string Point::getName() const 47 { 48 return "Point"; 49 50 } // end function getName 51 52 // override pure virtual function print: output Point object 53 void Point::print() const 54 { 55 cout << '[' << getX() << ", " << getY() << ']'; 56 57 } // end function print Must override pure virtual functions getName and print.

48 Outline 48 circle.h (1 of 2) 1 // Fig. 10.16: circle.h 2 // Circle class contains x-y coordinate pair and radius. 3 #ifndef CIRCLE_H 4 #define CIRCLE_H 5 6 #include "point.h" // Point class definition 7 8 class Circle : public Point { 9 10 public: 11 12 // default constructor 13 Circle( int = 0, int = 0, double = 0.0 ); 14 15 void setRadius( double ); // set radius 16 double getRadius() const; // return radius 17 18 double getDiameter() const; // return diameter 19 double getCircumference() const; // return circumference 20 virtual double getArea() const; // return area 21 22 // return name of shape (i.e., "Circle") 23 virtual string getName() const; 24 25 virtual void print() const; // output Circle object

49 Outline 49 circle.h (2 of 2) 26 27 private: 28 double radius; // Circle's radius 29 30 }; // end class Circle 31 32 #endif

50 Outline 50 circle.cpp (1 of 3) 1 // Fig. 10.17: circle.cpp 2 // Circle class member-function definitions. 3 #include 4 5 using std::cout; 6 7 #include "circle.h" // Circle class definition 8 9 // default constructor 10 Circle::Circle( int xValue, int yValue, double radiusValue ) 11 : Point( xValue, yValue ) // call base-class constructor 12 { 13 setRadius( radiusValue ); 14 15 } // end Circle constructor 16 17 // set radius 18 void Circle::setRadius( double radiusValue ) 19 { 20 radius = ( radiusValue < 0.0 ? 0.0 : radiusValue ); 21 22 } // end function setRadius 23

51 Outline 51 circle.cpp (2 of 3) 24 // return radius 25 double Circle::getRadius() const 26 { 27 return radius; 28 29 } // end function getRadius 30 31 // calculate and return diameter 32 double Circle::getDiameter() const 33 { 34 return 2 * getRadius(); 35 36 } // end function getDiameter 37 38 // calculate and return circumference 39 double Circle::getCircumference() const 40 { 41 return 3.14159 * getDiameter(); 42 43 } // end function getCircumference 44

52 Outline 52 circle.cpp (3 of 3) 45 // override virtual function getArea: return area of Circle 46 double Circle::getArea() const 47 { 48 return 3.14159 * getRadius() * getRadius(); 49 50 } // end function getArea 51 52 // override virutual function getName: return name of Circle 53 string Circle::getName() const 54 { 55 return "Circle"; 56 57 } // end function getName 58 59 // override virtual function print: output Circle object 60 void Circle::print() const 61 { 62 cout << "center is "; 63 Point::print(); // invoke Point's print function 64 cout << "; radius is " << getRadius(); 65 66 } // end function print Override getArea because it now applies to Circle.

53 Outline 53 cylinder.h (1 of 2) 1 // Fig. 10.18: cylinder.h 2 // Cylinder class inherits from class Circle. 3 #ifndef CYLINDER_H 4 #define CYLINDER_H 5 6 #include "circle.h" // Circle class definition 7 8 class Cylinder : public Circle { 9 10 public: 11 12 // default constructor 13 Cylinder( int = 0, int = 0, double = 0.0, double = 0.0 ); 14 15 void setHeight( double ); // set Cylinder's height 16 double getHeight() const; // return Cylinder's height 17 18 virtual double getArea() const; // return Cylinder's area 19 virtual double getVolume() const; // return Cylinder's volume 20

54 Outline 54 cylinder.h (2 of 2) 21 // return name of shape (i.e., "Cylinder" ) 22 virtual string getName() const; 23 24 virtual void print() const; // output Cylinder 25 26 private: 27 double height; // Cylinder's height 28 29 }; // end class Cylinder 30 31 #endif

55 Outline 55 cylinder.cpp (1 of 3) 1 // Fig. 10.19: cylinder.cpp 2 // Cylinder class inherits from class Circle. 3 #include 4 5 using std::cout; 6 7 #include "cylinder.h" // Cylinder class definition 8 9 // default constructor 10 Cylinder::Cylinder( int xValue, int yValue, double radiusValue, 11 double heightValue ) 12 : Circle( xValue, yValue, radiusValue ) 13 { 14 setHeight( heightValue ); 15 16 } // end Cylinder constructor 17 18 // set Cylinder's height 19 void Cylinder::setHeight( double heightValue ) 20 { 21 height = ( heightValue < 0.0 ? 0.0 : heightValue ); 22 23 } // end function setHeight

56 Outline 56 cylinder.cpp (2 of 3) 24 25 // get Cylinder's height 26 double Cylinder::getHeight() const 27 { 28 return height; 29 30 } // end function getHeight 31 32 // override virtual function getArea: return Cylinder area 33 double Cylinder::getArea() const 34 { 35 return 2 * Circle::getArea() + // code reuse 36 getCircumference() * getHeight(); 37 38 } // end function getArea 39 40 // override virtual function getVolume: return Cylinder volume 41 double Cylinder::getVolume() const 42 { 43 return Circle::getArea() * getHeight(); // code reuse 44 45 } // end function getVolume 46

57 Outline 57 cylinder.cpp (3 of 3) 47 // override virtual function getName: return name of Cylinder 48 string Cylinder::getName() const 49 { 50 return "Cylinder"; 51 52 } // end function getName 53 54 // output Cylinder object 55 void Cylinder::print() const 56 { 57 Circle::print(); // code reuse 58 cout << "; height is " << getHeight(); 59 60 } // end function print

58 Outline 58 fig10_20.cpp (1 of 5) 1 // Fig. 10.20: fig10_20.cpp 2 // Driver for shape, point, circle, cylinder hierarchy. 3 #include 4 5 using std::cout; 6 using std::endl; 7 using std::fixed; 8 9 #include 10 11 using std::setprecision; 12 13 #include 14 15 using std::vector; 16 17 #include "shape.h" // Shape class definition 18 #include "point.h" // Point class definition 19 #include "circle.h" // Circle class definition 20 #include "cylinder.h" // Cylinder class definition 21 22 void virtualViaPointer( const Shape * ); 23 void virtualViaReference( const Shape & ); 24

59 Outline 59 fig10_20.cpp (2 of 5) 25 int main() 26 { 27 // set floating-point number format 28 cout << fixed << setprecision( 2 ); 29 30 Point point( 7, 11 ); // create a Point 31 Circle circle( 22, 8, 3.5 ); // create a Circle 32 Cylinder cylinder( 10, 10, 3.3, 10 ); // create a Cylinder 33 34 cout << point.getName() << ": "; // static binding 35 point.print(); // static binding 36 cout << '\n'; 37 38 cout << circle.getName() << ": "; // static binding 39 circle.print(); // static binding 40 cout << '\n'; 41 42 cout << cylinder.getName() << ": "; // static binding 43 cylinder.print(); // static binding 44 cout << "\n\n"; 45

60 Outline 60 fig10_20.cpp (3 of 5) 46 // create vector of three base-class pointers 47 vector shapeVector( 3 ); 48 49 // aim shapeVector[0] at derived-class Point object 50 shapeVector[ 0 ] = &point; 51 52 // aim shapeVector[1] at derived-class Circle object 53 shapeVector[ 1 ] = &circle; 54 55 // aim shapeVector[2] at derived-class Cylinder object 56 shapeVector[ 2 ] = &cylinder; 57 58 // loop through shapeVector and call virtualViaPointer 59 // to print the shape name, attributes, area and volume 60 // of each object using dynamic binding 61 cout << "\nVirtual function calls made off " 62 << "base-class pointers:\n\n"; 63 64 for ( int i = 0; i < shapeVector.size(); i++ ) 65 virtualViaPointer( shapeVector[ i ] ); 66 Create a vector of generic Shape pointers, and aim them at various objects. Function virtualViaPointer calls the virtual functions ( print, getName, etc.) using the base- class pointers. The types are dynamically bound at run-time.

61 Outline 61 fig10_20.cpp (4 of 5) 67 // loop through shapeVector and call virtualViaReference 68 // to print the shape name, attributes, area and volume 69 // of each object using dynamic binding 70 cout << "\nVirtual function calls made off " 71 << "base-class references:\n\n"; 72 73 for ( int j = 0; j < shapeVector.size(); j++ ) 74 virtualViaReference( *shapeVector[ j ] ); 75 76 return 0; 77 78 } // end main 79 80 // make virtual function calls off a base-class pointer 81 // using dynamic binding 82 void virtualViaPointer( const Shape *baseClassPtr ) 83 { 84 cout getName() << ": "; 85 86 baseClassPtr->print(); 87 88 cout getArea() 89 getVolume() 90 << "\n\n"; 91 92 } // end function virtualViaPointer 93 Use references instead of pointers, for the same effect. Call virtual functions; the proper class function will be called at run-time.

62 Outline 62 fig10_20.cpp (5 of 5) 94 // make virtual function calls off a base-class reference 95 // using dynamic binding 96 void virtualViaReference( const Shape &baseClassRef ) 97 { 98 cout << baseClassRef.getName() << ": "; 99 100 baseClassRef.print(); 101 102 cout << "\narea is " << baseClassRef.getArea() 103 << "\nvolume is " << baseClassRef.getVolume() << "\n\n"; 104 105 } // end function virtualViaReference

63 Outline 63 fig10_20.cpp output (1 of 2) Point: [7, 11] Circle: center is [22, 8]; radius is 3.50 Cylinder: center is [10, 10]; radius is 3.30; height is 10.00 Virtual function calls made off base-class pointers: Point: [7, 11] area is 0.00 volume is 0.00 Circle: center is [22, 8]; radius is 3.50 area is 38.48 volume is 0.00 Cylinder: center is [10, 10]; radius is 3.30; height is 10.00 area is 275.77 volume is 342.12

64 Outline 64 fig10_20.cpp output (2 of 2) Virtual function calls made off base-class references: Point: [7, 11] area is 0.00 volume is 0.00 Circle: center is [22, 8]; radius is 3.50 area is 38.48 volume is 0.00 Cylinder: center is [10, 10]; radius is 3.30; height is 10.00 area is 275.77 volume is 342.12


Download ppt "Outline 1 Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü ++ Bilişim Enstitüsü."

Similar presentations


Ads by Google