C++ 实验三 继承和派生类 (附答案)

实验三  继承和派生类

实验目的和要求

    1.理解类的继承的概念,能够定义和使用类的继承关系。

    2.掌握派生类的声明与定义方法。

    3.熟悉公有派生和私有派生的访问特性。

    4.学习虚基类在解决二义性问题中的作用。

实验内容

1.先阅读下面的程序,分析程序运行的结果,然后再上机运行程序,验证自己分析的结果是否正确。

(1)

#include<iostream.h>

class A

{

public:

      A()

      {

             cout<<"A::A()called.\n";

      }

      ~A()

      {

             cout<<"A::~A()called.\n";

      }

};

class B:public A

{

public:

      B(inti)

      {

             cout<<"B::B()called.\n";

             buf=newchar[i];

      }

      ~B()

      {

             delete[]buf;

             cout<<"B:~B()called.\n";

      }

private:

      char*buf;

};

void main()

{

      Bb(10);

}

(2)

#include<iostream.h>

class A

{

public:

      A(inta,int b):x(a),y(b)

      {

             cout<<"Aconstructor..."<<endl;

      }

      voidAdd(int a,int b)

      {

             x+=a;

             y+=b;

      }

      voiddisplay()

      {

             cout<<"("<<x<<","<<y<<")"<<endl;

      }

      ~A()

      {

             cout<<"destructorA..."<<endl;

      }

private:

      intx,y;

};

class B:private A

{

private:

      inti,j;

      AAobj;

public:

      B(inta,int b,int c,int d):A(a,b),i(c),j(d),Aobj(1,1)

      {

             cout<<"Bconstructor..."<<endl;

      }

      voidAdd(int x1,int y1,int x2,int y2)

      {

             A::Add(x1,y1);

             i+=x2;

             j+=y2;

      }

      voiddisplay()

      {

             A::display();

             Aobj.display();

             cout<<"("<<i<<","<<j<<")"<<endl;

      }

      ~B()

      {

             cout<<"destructorB..."<<endl;

      }

};

void main()

{

      Bb(1,2,3,4);

      b.display();

      b.Add(1,3,5,7);

      b.display();

}

(3)

#include<iostream.h>

class A

{

public:

      A(inta):x(a)

      {

             cout<<"Aconstructor..."<<x<<endl;

      }

      intf()

      {

             return++x;

      }

      ~A()

      {

             cout<<"destructorA..."<<endl;

      }

private:

      intx;

};

class B:public virtual A

{

private:

      inty;

      AAobj;

public:

      B(inta,int b,int c):A(a),y(c),Aobj(c)

      {

             cout<<"Bconstructor..."<<y<<endl;

      }

      intf()

      {

             A::f();

             Aobj.f();

             return++y;

      }

      voiddisplay()

      {

             cout<<A::f()<<'\t'<<Aobj.f()<<'\t'<<f()<<endl;

      }

      ~B()

      {

             cout<<"destructorB..."<<endl;

      }

};

class C:public B

{

public:

      C(inta,int b,int c):B(a,b,c),A(0)

      {

             cout<<"Cconstructor..."<<endl;

      }

};

class D:public C,virtual public A

{

public:

      D(inta,int b,int c):C(a,b,c),A(c)

      {

             cout<<"Dconstructor..."<<endl;

      }

      ~D()

      {

             cout<<"destructorD..."<<endl;

      }

};

void main()

{

      Dd(7,8,9);

      d.f();

      d.display();

}

(4)

#include<iostream.h>

class Base1

{

public:

      Base1()

      {

             cout<<"classBase1!"<<endl;

      }

};

class Base2

{

public:

      Base2()

      {

             cout<<"classBase2!"<<endl;

      }

};

class Level1:public Base2,virtual publicBase1

{

public:

      Level1()

      {

             cout<<"classLevel1!"<<endl;

      }

};

class Level2:public Base2,virtual publicBase1

{

public:

      Level2()

      {

             cout<<"classLevel2!"<<endl;

      }

};

class TopLevel:public Level1,virtual publicLevel2

{

public:

      TopLevel()

      {

             cout<<"classTopLevel!"<<endl;

      }

};

void main()

{

      TopLevelobj;

}

2.调试下列程序,并对程序进行修改后再调试,指出调试中的出错原因。

#include <iostream.h>

class A

{

public:

      voidseta(int i)

      {     a=i;  }

      intgeta()

      {     return a;  }

public:

      inta;

};

class B:public A

{

public:

      voidsetb(int i)

      {     b=i; }

      intgetb()

      {     return b;  }

      voidshow()

      {     cout<<"A::a="<<a<<endl;     } //语句9

public:

      intb;

};

void main()

{

      Bbb;                                                       //语句1

      bb.seta(6);                                                //语句2

      bb.setb(3);                                                //语句3

      bb.show();                                                //语句4

      cout<<"A::a="<<bb.a<<endl;                            //语句5

      cout<<"B::b="<<bb.b<<endl;                            //语句6

      cout<<"A::a="<<bb.geta()<<endl;               //语句7

      cout<<"B::b="<<bb.getb()<<endl;                     //语句8

}

按下列要求对程序进行修改,然后调试,对出现的错误分析其原因。

(1)将派生类B的继承方式改为private时,会出现哪些错误和不正常现象?为什么?

(2)将派生类B的继承方式改为protected时,会出现哪些错误和不正常现象?为什么?

(3)将派生类B的继承方式恢复为public后,再将类A中数据成员a的访问权限改为private时,会出现哪些错误和不正常现象?为什么?

(4)派生类B的继承方式仍为public,将类A中数据成员a的访问权限改为protected时,会出现哪些错误和不正常现象?为什么?

    3.定义一个类MyArray,类中可以存放一组整数。类的定义如下:

            class MyArray

            {

            public:

                MyArray(int len);          //初始化,动态申请存储空间

                ~MyArray();                 //对象结束时,释放所占用的存储空间

                void Input();                 //输入数据

                void Display();              //显示数据

            protected:

              int *alist;     //指向动态申请的一组空间

              int length;    //整数的个数

            };

        基类中有构造函数、析构函数、输入数据和输出数据的函数。

       (1)实现该类并设计主函数对所定义的类进行调试。下同。

(2)由MyArray派生一个类SortArray,在该类中定义函数实现排序功能。

(3)由MyArray派生一个类ReArray,在该类中定义函数实现数据前后逆转功能。

(4)由MyArray派生一个类AverArray,在该类中定义函数Aver求解类中所有数据的平均值。

(5)定义NewArray类,同时继承了SortArray, ReArray和AverArray,使得NewArray类的对象同时具有排序、逆转和求平均值的功能。在继承的过程中声明为虚基类,体会虚基类在解决二义性问题中的作用。




参考答案(非权威,仅仅是我自己的理解,如有错误,欢迎批评指正!)


第二题:
(1)主函数中语句2、语句5、语句7出现错误。因为B采用私有继承,所以A中的公有函数在B中是变为私有函数,主函数中不能访问B中的私有函数。

(2)主函数中语句2、语句5、语句7出现错误。因为B采用保护继承,所以A中的公有函数在B中是变为保护函数,主函数中不能访问B中的保护函数。

(3)主函数中语句9、语句5出现错误。因为B采用公有继承,所以B中和主函数中不能访问A中的私有函数。

(4)主函数中语句5出现错误。因为B采用公有继承,所以A中保护成员在B中仍为保护成员,B中可以访问A中的保护成员,但不能通过类的对象访问类的保护成员。

第三题:
(1)
#include <iostream.h>
class Myarray
{
protected:
		int *alist;		 //指向动态申请的一组空间
		int length;		//整数的个数
public:
    Myarray(int len)		//初始化,动态申请存储空间
    {
			alist=new int[len];
			length=len;
	}
    ~Myarray()			//对象结束时,释放所占用的存储空间
    {
			delete(alist);
			cout<<endl<<"再见!"<<endl;
	}
    void Input()			//输入数据
    {
        int i;
        for(i=0;i<length;i++)
        {
				cout<<"输入数组的第"<<i+1<<"个元素:";
				 cin>>alist[i];
		}
    }
    void Display()		//显示数据
    {
			cout<<'\n'<<"数组的成员为:"<<'\n';
			for(int i=0;i<length;i++)
		 cout<<alist[i]<<' ';
		cout<<endl;
    }
};

void main()
{
		Myarray A(3);
		A.Input();
		A.Display();
}

(2)
#include <iostream.h>
class Myarray
{
protected:
		int *alist;		 //指向动态申请的一组空间
		int length;		//整数的个数
public:
    Myarray(int len)		//初始化,动态申请存储空间
    {
			alist=new int[len];
			length=len;
			cout<<"Myarray 初始化..."<<endl;
	}
    ~Myarray()			//对象结束时,释放所占用的存储空间
    {
			delete(alist);
			cout<<endl<<"再见!"<<endl;
	}
    void Input()			//输入数据
    {
        int i;
        for(i=0;i<length;i++)
        {
				cout<<"输入数组的第"<<i+1<<"个元素:";
				 cin>>alist[i];
		}
    }
    void Display()		//显示数据
    {
			cout<<'\n'<<"本数组成员依次为:"<<'\n';
			for(int i=0;i<length;i++)
		 cout<<alist[i]<<' ';
		cout<<endl;
    }
};

class SortArray:virtual public Myarray
{
public:
		    SortArray(int n):Myarray(n)
		{
  			cout<<"SortArray 初始化..."<<endl;
		}
		void sort()
		{
		    int i,j,k,temp;
		     for(i=0;i<length-1;i++)
		    {
				k=i;
				for(j=i+1;j<length;j++)
				     if(alist[j]<alist[k])
				     {
						 k=j;
				     }
				  temp=alist[i];
				  alist[i]=alist[k];
				  alist[k]=temp;
			 }
		}
};

void main()
{
		SortArray B(4);
		B.Input();
		B.Display();
		cout<<endl<<"现在将数组内成员进行从小到大排序:"<<endl;
		B.sort();
		B.Display();
}

(3)
#include <iostream.h>
class Myarray
{
protected:
		int *alist;		 //指向动态申请的一组空间
		int length;		//整数的个数
public:
    Myarray(int len)		//初始化,动态申请存储空间
    {
			alist=new int[len];
			length=len;
			cout<<"Myarray 初始化..."<<endl;
	}
    ~Myarray()			//对象结束时,释放所占用的存储空间
    {
			delete(alist);
			cout<<endl<<"再见!"<<endl;
	}
    void Input()			//输入数据
    {
        int i;
        for(i=0;i<length;i++)
        {
				cout<<"输入数组的第"<<i+1<<"个元素:";
				 cin>>alist[i];
		}
    }
    void Display()		//显示数据
    {
			cout<<'\n'<<"本数组成员依次为:"<<'\n';
			for(int i=0;i<length;i++)
		 cout<<alist[i]<<' ';
		cout<<endl;
    }
};

class ReArray: virtual public Myarray
{
public:
    ReArray(int n):Myarray(n)
		{
  			cout<<"ReArray 初始化..."<<endl;
		}
    void turn()
		{
			int i,t,j=length-1;
				for(i=0;i<length/2;i++,j--)
				{
						t=alist[i];
						alist[i]=alist[j];
						alist[j]=t;
				}
		}
};

void main()
{
		ReArray B(4);
		B.Input();
		B.Display();
		cout<<endl<<"现在将数组内成员进行前后逆转:"<<endl;
		B.turn();
		B.Display();
}

(4)
#include <iostream.h>
class Myarray
{
protected:
		int *alist;		 //指向动态申请的一组空间
		int length;		//整数的个数
public:
    Myarray(int len)		//初始化,动态申请存储空间
    {
			alist=new int[len];
			length=len;
			cout<<"Myarray 初始化..."<<endl;
	}
    ~Myarray()			//对象结束时,释放所占用的存储空间
    {
			delete(alist);
			cout<<endl<<"再见!"<<endl;
	}
    void Input()			//输入数据
    {
        int i;
        for(i=0;i<length;i++)
        {
				cout<<"输入数组的第"<<i+1<<"个元素:";
				 cin>>alist[i];
		}
    }
    void Display()		//显示数据
    {
			cout<<'\n'<<"本数组成员依次为:"<<'\n';
			for(int i=0;i<length;i++)
		 cout<<alist[i]<<' ';
		cout<<endl;
    }
};

class AverArray:virtual public Myarray
{
public:
		AverArray(int n):Myarray(n)
		{
  			cout<<"AverArray 初始化..."<<endl;
		}
    float getaver()
    {
			float aver;
	        int sum=0,i;
		     for(i=0;i<length;i++)
		     sum+=alist[i];
		     aver=(float)sum/length;
		     return aver;
    }
};

void main()
{
		AverArray B(3);
		B.Input();
		B.Display();
		cout<<endl<<"数组内成员的平均值为:";
		cout<<B.getaver()<<' '<<endl;
}

(5)
#include <iostream.h>
class Myarray
{
protected:
		int *alist;		 //指向动态申请的一组空间
		int length;		//整数的个数
public:
    Myarray(int len)		//初始化,动态申请存储空间
    {
			alist=new int[len];
			length=len;
			cout<<"Myarray 初始化..."<<endl;
	}
    ~Myarray()			//对象结束时,释放所占用的存储空间
    {
			delete(alist);
			cout<<endl<<"再见!"<<endl;
	}
    void Input()			//输入数据
    {
        int i;
        for(i=0;i<length;i++)
        {
				cout<<"输入数组的第"<<i+1<<"个元素:";
				 cin>>alist[i];
		}
    }
    void Display()		//显示数据
    {
			cout<<endl<<"本数组成员依次为:";
			for(int i=0;i<length;i++)
		 cout<<alist[i]<<' ';
		cout<<endl;
    }
};

class SortArray:virtual public Myarray	//从小到大排序
{
public:
		    SortArray(int n):Myarray(n)
		{
  			cout<<"SortArray 初始化..."<<endl;
		}
		void get_SortArray()
		{
		    int i,j,k,temp;
		     for(i=0;i<length-1;i++)
		    {
				k=i;
				for(j=i+1;j<length;j++)
				     if(alist[j]<alist[k])
				     {
						 k=j;
				     }
				  temp=alist[i];
				  alist[i]=alist[k];
				  alist[k]=temp;
			 }
		}
};

class ReArray: virtual public Myarray	//前后逆转
{
public:
    ReArray(int n):Myarray(n)
		{
  			cout<<"ReArray 初始化..."<<endl;
		}
    void get_ReArray()
		{
			int i,t,j=length-1;
				for(i=0;i<length/2;i++,j--)
				{
						t=alist[i];
						alist[i]=alist[j];
						alist[j]=t;
				}
		}
};

class AverArray:virtual public Myarray	//求平均值
{
public:
		AverArray(int n):Myarray(n)
		{
  			cout<<"AverArray 初始化..."<<endl;
		}
    float get_AverArray()
    {
			float aver;
	        int sum=0,i;
		     for(i=0;i<length;i++)
		     sum+=alist[i];
		     aver=(float)sum/length;
		     return aver;
    }
};

class NewArray:public SortArray,public ReArray,public AverArray
{
public:
		NewArray(int n):SortArray(n),ReArray(n),AverArray(n),Myarray(n)
		{
  			cout<<"NewArray 初始化..."<<endl;
		}
		void fun()
		{
		cout<<endl<<"数组成员的平均值为:";
		cout<<AverArray::get_AverArray()<<' '<<endl;		

		cout<<endl<<"现在将数组成员进行前后逆转:"<<endl;
		ReArray::get_ReArray();
		Myarray::Display();

		cout<<endl<<"现在将数组成员进行从小到大排序:"<<endl;
		SortArray::get_SortArray();
	    Myarray::Display();
		}
};

void main()
{
		NewArray B(3);
		B.Input();
		B.Display();
		B.fun();
}




  • 1
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 在C++中,继承是一种重要的面向对象编程概念,它允许我们定义一个新的类,该类继承现有类的所有属性和方法。派生类继承类的子类,它可以使用继承类的所有属性和方法,并且可以添加自己的属性和方法。 在实验中,我们学习了C++继承派生类的相关知识,包括: 1. 继承类的访问控制:公有继承、私有继承和保护继承。 2. 多重继承:一个派生类可以同时继承多个基类。 3. 虚函数和纯虚函数:虚函数是在基类中定义的函数,可以在派生类中重写;纯虚函数是没有实现的虚函数,必须在派生类中实现。 4. 虚函数表和虚函数指针:虚函数表是用于存储虚函数地址的表,虚函数指针指向虚函数表。 通过实验,我们能够更深入地理解C++继承派生类的概念和用,能够更加灵活地使用面向对象编程思想来设计和实现程序。 ### 回答2: 继承是面向对象编程中的重要概念之一,它允许我们在已有类的基础上创建新的类,新类可以继承并拥有已有类的属性和方法。通过继承,可以减少代码的重复性,并且使代码更加可维护和扩展。 在进行派生类实验的过程中,我深刻体会到了继承的重要性和灵活性。首,通过定义一个基类,我可以将一些通用的属性和方法抽象出来,避免在每个派生类中都重复定义。这为程序的整体结构设计提供了便捷。 其次,派生类可以在继承基类的基础上进行扩展,添加新的属性和方法。这种灵活性使得派生类在满足基本功能的同时,也能根据具体需求进行定制化开发。例如,在一个动物类的基础上,我可以派生出猫、狗等具体的动物子类,它们各自有着自己的特点和行为。 另外,继承还支持多层次的派生关系。我可以从一个派生类中再派生出新的派生类,这样可以形成类的层次结构。这种层次化设计可以更好地组织代码,使得代码更加清晰可读。 通过这次实验,我进一步理解了继承性与派生类的概念,学会了如何设计和使用继承关系。同时,我也认识到了继承关系的合理运用能够提高代码的效率和可维护性。继承不仅是面向对象编程的基础,也是实现代码重用和扩展的重要工具之一。在今后的编程实践中,我将更加灵活地运用继承,提高代码的质量和可扩展性。 ### 回答3: 继承性是面向对象编程中的一个重要特性,指的是子类能够继承父类的属性和方法。通过继承,子类可以重用父类的代码,并且可以在此基础上进行扩展和修改。 派生类是指通过继承父类而创建的新类。在派生类中,可以通过重写父类的方法,改变其行为,实现多态性。派生类还可以新增自己的成员变量和成员方法,以满足自身的特殊需求。 在实验中,我们通过创建父类和派生类的关系,研究了继承性和派生类的特性。 通过继承,我们可以将通用的属性和方法放在父类中,让子类共享这些代码。这样可以提高代码的重用性和可维护性。同时,当需要对父类中的方法进行修改时,只需在子类中进行重写,不会对其他子类造成影响。 在派生类中,我们可以根据需要重写父类的方法,改变其行为。这使得可以根据实际情况来实现多态性,同一个方法在不同的派生类中可能表现出不同的行为。 派生类还可以新增自己的成员变量和成员方法。通过这样的方式,可以为子类添加独特的功能,以满足特定的需求。 继承性和派生类是面向对象编程中非常重要的概念。通过合理运用这两个特性,可以使代码更加模块化和可扩展,提高代码的复用性和可维护性。同时,派生类的特性也使得面向对象编程更加灵活,可以根据实际需求进行扩展和修改。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值