联系方式

  • QQ:99515681
  • 邮箱:99515681@qq.com
  • 工作时间:8:00-21:00
  • 微信:codinghelp

您当前位置:首页 >> Python编程Python编程

日期:2020-10-08 11:46

3.4 - Simple Inheritance

Exercise 1: Colon Syntax

The colon syntax can improve the performance of constructors. To test this, make sure that

you print some text in the point’s constructors, destructor and also the assignment operator.

Now, execute the following code in the test program and count the number of point

constructor, destructor and assignment calls:

Line l;

Now, change the constructors in the Line class to use the colon syntax to set the start- and

end-point data members and run the test program again. Is the number of point constructor,

destructor and assignment calls less than before?

Apply the colon syntax also for the Point class constructors and if applicable also for

the Circle class.

Exercise 2: Creating Shape Base Class

It can be useful to create a hierarchy of related classes using base- and derived classes.

• Classes are related (same family)

• Common data and functionality can be put in a base class.

• You can work with derived classes as if it is the base class.

In this exercise we are going to transform the Point and Line class into a Shape hierarchy as

shown in Figure 1.

+ToString()

Point

+ID()

+ToString()

-m_id

Shape

+ToString()

Line

+ToString()

Circle

Figure 1: Shape Hierarchy

First create a Shape base class.

• Add a source- and header file for a Shape class.

• Add a data member for an id number of type int.

• Add a default constructor that initializes the id using a random number. You can use

the rand() function from the “stdlib.h” header file.

• Add a copy constructor that copies the id member.

• Add an assignment operator that copies the id member.

• Add a ToString() function that returns the id as string e.g. “ID: 123”.

• Add an ID() function the retrieve the id of the shape.

Next the Point and Line classes (and the Circle class if applicable) must derive from Shape.

• Add the Shape class in the inheritance list of the Point, Line and optionally

the Circle class.

• The constructors of the Point, Line and optionally the Circle class should call the

appropriate constructor in the Shapebase class.

• The assignment operator should call the assignment operator of the Shape base

class. Otherwise the shape data will not be copied.

• Finally add code to the main program to test inheritance:

Shape s; // Create shape.

Point p(10, 20); // Create point.

Line l(Point(1,2), Point(3, 4)); // Create line.

cout<<s.ToString()<<endl; // Print shape.

cout<<p.ToString()<<endl; // Print point.

cout<<l.ToString()<<endl; // Print line

cout<<"Shape ID: "<<s.ID()<<endl; // ID of the shape.

cout<<"Point ID: "<<p.ID()<<endl; // ID of the point. Does this work?

cout<<"Line ID: "<<l.ID()<<endl; // ID of the line. Does this work?

Shape* sp; // Create pointer to a shape variable.

sp=&p; // Point in a shape variable. Possible?

cout<<sp->ToString()<<endl; // What is printed?

// Create and copy Point p to new point.

Point p2;

p2=p;

cout<<p2<<", "<<p2.ID()<<endl; // Is the ID copied if you do not call

// the base class assignment in point?

Answer the questions in the comments of the code above.

3.5 - Polymorphism

Exercise 1: Polymorphic ToString() Function

The ToString() functions of Point and Line override the ToString() from the Shape base class.

We saw that we could put aPoint in a Shape* variable. But when calling

the ToString() method on the Shape* variable, the function in Shape was called instead the

one in Point. To make the compiler generate the required code to find out what type of object

the Shape*variable is actually pointing to so it can call the right version; we need to declare

the function as virtual.

Thus declare the ToString() function in the Shape class as virtual and test the program again.

Is the ToString() function of Point called when you use a Shape* that contains a Point now?

Exercise 2: Calling Base Class Functionality

The ToString() function of the Shape class is overridden in the derived classes. But for the

derived class it is still possible to use the base class functionality. In the ToString() function

of Point and Line we also want to incorporate the ID from theShape base class.

• In the ToString() method of Point, call the ToString() method of the Shape base class:

std::string s=Shape::ToString();

• Append the shape description string to the point description string before returning.

• Do this also for the ToString() function in the Line class (and Circle class).

• Test the application again. Is now the ID printed when printing a point or line?

Exercise 3: Virtual Destructors

When objects are removed from memory, the destructor is called. When a derived class

destructor is called, it will automatically call the base class destructor. But when you have

pointers to a base class, deleting objects might not be done correctly.

If not done already, print some text in the destructors of the Shape, Point and Line classes.

Then test the following code:

Shape* shapes[3];

shapes[0]=new Shape;

shapes[1]=new Point;

shapes[2]=new Line;

for (int i=0; i!=3; i++) delete shapes[i];

Will the proper destructors (including the destructor of the Shape base class) be called?

In this case, the derived class destructor will only be called when the destructor is

declared virtual in the base class. Do this in the Shape class and run the code again. Are the

proper destructors called now?

Exercise 4: Abstract Functions

Sometimes functions in the base class are only there to be overridden in the derived class.

Assume that you want to draw all the shapes using the following code:

Shape* shapes[10];

shapes[0]=new Line;

shapes[1]=new Point;

...

shapes[9]=new Line(Point(1.0, 2.5), Point(3.4, 5.2));

for (int i=0; i!=10; i++) shapes[i]->Draw();

for (int i=0; i!=10; i++) delete shapes[i];

Create the Draw() function in the Shape base class and override it in the derived classes

(point, line and if present the circle class). Simulate drawing by just printing some text.

What implementation did you give the Draw() function in Shape? Shape is just an abstraction

to work with various kinds of shapes like lines and circles. Shapes don’t have a physical

appearance. Therefore its Draw() function will have an empty implementation. But better is

to give it no implementation at all by making it a pure virtual member function:

virtual void Draw()=0;

Do this in your code. Try to create an instance of the Shape class. Is this possible?

Now the Shape class is really an abstraction. You don’t make shape instances but you can

still create shape pointers that point to concrete shapes like point and line. The Shape class is

now an abstract base class.

+ToString()

+Draw()

Point

+ID()

+ToString()

+Draw()

-m_id

Shape

+ToString()

+Draw()

Line

+ToString()

+Draw()

Circle

Figure 2: Abstract Shape base class with Draw() pure virtual member function

Exercise 5: Template Method Pattern

In this exercise we are going to create a Print() function that is printing the shape information

to the cout object. ThePrint() function can use the ToString() to obtain the string to print.

You see that the implementation of Print() is in each derived class largely the same;

calling ToString() and sending the result to cout. Since the ToString() function is

polymorphic, we can use the polymorphic behavior in the Print() function of Shape.

Thus:

• Add a Print() function to the Shape class.

• In this function, call the ToString() function and send the result to the cout object.

• In the test program, create a point and line object and call the Print() function. Does it

print the right information even when point and line do not have the Print() function?

You have now created a function for the base class (Print()) that does all the functionality

common to all derived classes. Only the part of that function that is different for each derived

class is delegated to a polymorphic function (ToString()). This mechanism is an often used

design pattern called the “Template Method Pattern”.

3.6 - Exception Handling

Exercise 1: Bounds Checking Array

In the array class we created previously, the bounds checking was very basic. There was no

error generated, but setting an element was ignored or the first element was returned.

Obviously you want to know if there was an out of bounds error. This is possible by

exception handling.

Change the Array class to throw exceptions:

• In the GetElement(), SetElement() and index operator throw -1 if the index was too

small or too big.

• In the main program, create an Array object and access an element that does not exist.

Run the program. What happens?

• The exception must be caught, so place the code inside a try ... catch block that

catches an int.

• In the catch handler, print an error message.

Exercise 2: Exception Objects

Throwing an int is an easy solution. But exception handling is also object oriented and allows

us to throw an object.

In this exercise we will create an exception hierarchy with an ArrayException base class and

an OutOfBoundsExceptionderived class as shown in Figure 3:

• You can implement both exception classes in the header file for simplicity.

• Give the ArrayException an abstract GetMessage() function that returns a std::string.

• Give the OutOfBoundsException class a constructor with an int as argument that

indicates the erroneous array index and store it in a data member.

• Override the GetMessage() function and let the implementation return a message

string saying the given index is out of bounds.

• In the Array class, throw now a OutOfBoundsException object instead of an integer.

• Change the main program so that it catches the ArrayException base class and uses

the GetMessage() function to display an error message.

+SetElement()

+GetElement()

+operator []()

-m_data

-m_size

Array

T

+GetMessage()

ArrayException

+GetMessage()

-m_index

OutOfBoundsException

+main()

MainProgram catches

uses throws

Figure 3: Array Exception


版权所有:编程辅导网 2021 All Rights Reserved 联系方式:QQ:99515681 微信:codinghelp 电子信箱:99515681@qq.com
免责声明:本站部分内容从网络整理而来,只供参考!如有版权问题可联系本站删除。 站长地图

python代写
微信客服:codinghelp