编程语言
首页 > 编程语言> > c++第七课—继承和派生

c++第七课—继承和派生

作者:互联网

继承方式与权限问题

class 子类 :继承方式 父类

 

#include <iostream>
#include <string>
using namespace std;
//父类
class parent
{
public:
	int a;
    //接口
    int& getNum()
    {
        return c;
    }
protected:
	int b;
private:
	int c;
};
//公有继承
class son1 :public parent
{
public:
	void func()
	{
		a = 10;//父类中的公共权限成员,到子类中依然是 公共权限
		b = 20;//父类中的保护权限成员,到子类中依然是 保护权限
		//c = 30;//父类中的私有权限成员,子类访问不到

	}
};
//保护继承
class son2 :protected parent
{
public:
	void func()
	{
		a = 100;//父类中的公共权限成员,到子类中是 保护权限
		b = 200;//父类中的保护权限成员,到子类中依然是 保护权限
		//c = 30;//父类中的私有权限成员,子类访问不到
        //但是可以通过接口访问私有
        cout<<getNum()<<endl;
	}
};
//私有继承
class son3 :private parent
{
public:
	void func()
	{
		a = 1000;//父类中的公共权限成员,到子类中是 私有权限
		b = 2000;//父类中的保护权限成员,到子类中是 私有权限
		//c = 30;//父类中的私有权限成员,子类访问不到
	}
};
//孙子
class GrandSon :public son3
{
public:
	void func()
	{
		//a = 10000;//到了son3中 a,b变成私有,即使是儿子,也是访问不到
		//b = 20000;
	}
};
void test1()
{
	son1 s1;
	s1.a = 10;
	//s1.b = 20;//到son1中b是保护权限  类外访问不到
}
void test2()
{
	son2 s2;
	//s2.a = 10;//到son2中a是保护权限  类外访问不到
	//s1.b = 20;//到son2中b是保护权限  类外访问不到
}
void test3()
{
	son3 s3;
	//s3.a = 10;//到son2中a是私有权限  类外访问不到
	//s3.b = 20;//到son2中b是私有权限  类外访问不到
}
int main()
{
	return 0;
}

①打开“VS 2019的开发人员命令提示符”(“选择Developer Command Prompt for VS2019”)工具②输入"cd "文件所在路径③输入"dir"④输入"cl /d1 reportSingleClassLayout子类名 .cpp文件名"就可以查看了

 

class A
{
public:
	int a;
protected:
	int b;
private:
	int c;
};
class B : public  A
{
public:

protected:
	int d;
};

cout << sizeof(B) << endl;//输出结果:16
#include <iostream>
#include <string>
using namespace std;
//父类
class Common
{
public:
	void header()
	{
		cout << "首页,公开课,登陆,注册...(公共头部)" << endl;
	}
	void footer()
	{
		cout << "帮助中心,交流合作,站内地图...(公共底部)" << endl;
	}
	void left()
	{
		cout << "Java,Python,C++,...(公共分类列表)" << endl;
	}
};
class Java :public Common
{
public:
	void content()
	{
		cout << "Java学科视频"<<endl;
	}
};
class Python :public Common
{
public:
	void content()
	{
		cout << "Python学科视频" << endl;
	}
};
void test1()
{
	cout << "Java下载视频页面如下:" << endl;
	Java ja;
	ja.header();
	ja.footer();
	ja.left();
	ja.content();
}
void test2()
{
	cout << "Python下载视频页面如下:" << endl;
	Python py;
	py.header();
	py.footer();
	py.left();
	py.content();
}
int main()
{
	test1();
	test2();
	return 0;
}

 

继承中的构造函数

单继承:只有一个父类                        多继承:两个或者两个以上的父类

单继承:先构造父类,再构造子类,析构的顺序与构造顺序相反

多继承:构造顺序和继承顺序一致,析构相反;任何构造顺序问题都和初始化参数列表无关

//继承的属性一直都在
class A
{
public:
	A(int a):a(a){}
	int a;
};

class B:public A
{
public:
	B(int a,int b) :A(a),b(b){}
	int b;
};

class C:public B
{
public:
	C(int a,int b,int c):B(a,b),c(c) {}
	int c;
};

class D :public C
{
public:
	D(int a, int b, int c,int d) :C(a,b,c), d(d) {}
	int d;
};
//单继承
#include <iostream>
#include <string>
using namespace std;
//父类
class Parent
{
public:
	Parent() { cout << "父类无参构造函数" << endl; }
	Parent(string Fname,string Sname):Fname(Fname),Sname(Sname){}

	
protected:
	string Fname;
	string Sname;
};
//子类
class Son :public Parent
{
public:
	Son() { cout << "子类无参构造函数" << endl; } //指针写法,父类必须存在无参的构造函数,缺省也可以
	Son(string Fname, string Sname, string sonSname):Parent(Fname,Sname)
	{
		//自己的属性用什么办法初始化都行
		this->sonFname = Fname;
		this->sonSname = sonSname;
	}
	void print()
	{
		cout << "父亲:" << Fname + Sname << endl;
		cout << "儿子:" << sonFname + sonSname << endl;
	}
protected:
	string sonFname;
	string sonSname;
};

int main()
{
	Son s1;//子类构造对象,优先调用父类构造函数
	Son son1("李", "狗", "猪");
	son1.print();
	return 0;
}

 

//多继承
#include <iostream>
#include <string>
using namespace std;
//父类1
class MM
{
public:
	MM() = default;
	MM(string mmFname, string mmSname)
	{
		this->mmFname = mmFname;
		this->mmSname = mmSname;
	}
protected:
	string mmFname;
	string mmSname;
};
//父类2
class GG
{
public:
	//GG() = default;
	GG(string ggFname, string ggSname)
	{
		this->ggFname = ggFname;
		this->ggSname = ggSname;
	}
protected:
	string ggFname;
	string ggSname;
};
//子类
class Girl:public MM,public GG
{
public:
	//Girl(){}//需要两个父类都有无参构造函数
	//如果没有默认默认构造函数,需要用参数列表初始化,有则不需要
	Girl(string mmFname, string mmSname, string ggFname, string ggSname)
		:MM(mmFname,mmSname),GG(ggFname,ggSname)
	{
		girlFname = mmFname + ggFname;
		girlSname = mmSname + ggSname;
	}
	void print()
	{
		cout << "MM名字:" << mmFname + mmSname << endl;
		cout << "GG名字:" << ggFname + ggSname << endl;
		cout << "Girl名字:" << girlFname + girlSname << endl;
	}
protected:
	string girlFname;
	string girlSname;

};
int main()
{
	Girl k("洋", "子", "欧", "文");
	k.print();
	
	return 0;
}

 继承中同名问题

①子类对象可以直接访问到子类中同名成员

②子类对象加作用域可以访问到父类同名成员

③当子类与父类拥有同名的成员函数,子类会隐藏父类中同名成员函数,加作用域可以访问到父类中同名函数

#include <iostream>
#include <string>
using namespace std;
class MM
{
public:
	MM(string name,int age):name(name),age(age){}
	void print()
	{
		cout << "父类打印" << endl;
	}

protected:
	string name;
	int age;
};

class Girl :public MM
{
public:
	Girl(string name,int age):MM("父类",88),name(name), age(age) {}
	void print()
	{
		//不做特别处理,就近原则
		cout << name << " " << age << endl;
		//用类名限定,就可以打印父类的数据
		cout << MM::name << " " << MM::age << endl;
		MM::print();//函数也是一样不做特别处理也是就近原则,所以要加类名限定
	}

protected:
	string name;
	int age;
};

int main()
{
	//正常对象调用
	Girl m("子类",44);
	m.print();
	MM mm("mm", 11);
	mm.print();

	//正常指针调用
	//就近原则
	Girl* pG = new Girl("newGirl", 23);
	pG->print();
	pG->MM::print();//这样也可以
	MM* mG = new MM("newMM", 32);
	mG->print();

	//非正常的指针
	//允许子类对象初始化父类指针
	MM* pMM = new Girl("newGirl", 999);
	//那pMM的print是MM类还是Girl?
	//在没有virtual情况下,看指针类型,有则看赋值对象
	pMM->print();

	//父类对象初始化子类指针,不安全
	//Girl* pGG = new MM("newMM", 5550);//错误
	
	//引发中断
	//Girl* pgg = (Girl*)mG;
	//pgg->print();
	return 0;
}

菱形继承

#include <iostream>
#include <string>
using namespace std;
class A
{
public:
	A(int a):a(a){}

protected:
	int a;
};
class B :virtual public  A
{
public:
	B(int a,int b):A(a),b(b){}
	void print2()
	{
		cout << a << endl;
	}
protected:
	int b;
};
class C :virtual public  A
{
public:
	C(int a, int c) :A(a),c(c){}
protected:
	int c;
};
class D :public  B,public C
{
public:
	//菱形继承,必须调用祖父的构造函数
	D() :B(1, 2), C(3, 4), A(9) {}//和B类,C类无关
	void print()
	{
		//只有一份,所以打印都是一样的
		cout << a << endl;
		cout << B::a << endl;
		cout << C::a << endl;
		print2();//间接访问也是一样的
	}
protected:
	
};
int main()
{
	D dd;
	dd.print();
	
	return 0;
}

标签:派生,第七课,子类,继承,c++,class,int,父类,public
来源: https://blog.csdn.net/m0_61327008/article/details/122783547