C++学习笔记_03类的继承 2021-04-15

简介: C++学习笔记_03类的继承 2021-04-15
//C++学习笔记_03类的继承   友元类 
#include<cstring>
#include<iostream>
#define C_MAX 10
using namespace std;
//如果 课程这个类,只有学生才有,不能说单独定义一个课程对象
class CCourse
{
    friend class CStudent;
private:
    char cname[C_MAX][16]; //课程名称(多个课程,最多 C_MAX 门课程)
                //二维数组,可以表示一组字符串
                //cname[0] = "语文", cname[1] = "数学"
                //char *pname[C_MAX];
    unsigned int cnt;              //当前修几门课程
    unsigned int score[C_MAX];     //各门课程的得分
    CCourse() :cnt(0){}  //相当于构造的时候执行 cnt = 0
public:
    ~CCourse();
    bool addCourse(char szName[]);//新增一门课程
    bool setScore(char szName[], int score); //给某个课程打分
    void Print(); //打印分数
};
CCourse::~CCourse(){}
bool CCourse::addCourse(char szName[]){
    if (cnt == C_MAX) return false;
    for (int i = 0; i < cnt; i++)
        if (strcmp(szName, cname[i]) == 0) // if(!x) 等价于 if(x==0)
            return false;
    strcpy(cname[cnt],szName);
    score[cnt] = 0;
    cnt++;
    return true;
}
bool CCourse::setScore(char szName[], int score){
    for (int i = 0; i < cnt; i++)
        if (strcmp(szName, cname[i]) == 0) {// if(!x) 等价于 if(x==0)
            this->score[i] = score;
            return true;
        }
    return false;//没有找到相关课程
}
void CCourse::Print(){
    for (int i = 0; i < cnt; i++)
        cout << cname[i] << " :" << score[i] << endl;        
    cout << endl;
}
class CStudent
{
private:
    char name[16];
    unsigned int age;    
public:
    CCourse cour; //课程属性
    CStudent();
    CStudent(char szName[], unsigned int uiAge);
    ~CStudent();
};
CStudent::CStudent(){}
CStudent::CStudent(char szName[], unsigned int uiAge){
    strcpy(name,szName);
    age = uiAge;
}
CStudent::~CStudent(){}
void TestStudent()
{
    CStudent S1("Lily", 21);
    CStudent S2("Jame", 22);
    //CCourse cour; //直接这样定义,没有意义(这是谁的课程?)
                    //我们能不能不允许 直接定义 CCourse 对象?
                    //我们可以把 CCourse 的构造函数设为 private
    //这个时候,我实现了 CCourse 的隐藏,
    //但是,同时出现一个新的问题:CStudent 不能定义 CCourse 对象
    //怎么办? --》 在CCourse中,设置 CStudent 为 友元类
    S1.cour.addCourse("语文");
    S1.cour.addCourse("数学");
    S1.cour.addCourse("英语");
    S2.cour.addCourse("语文");
    S2.cour.addCourse("数学");
    S2.cour.addCourse("日语");
    S1.cour.setScore("语文", 98);
    S1.cour.setScore("英语", 88);
    S1.cour.setScore("数学", 68);
    S2.cour.setScore("语文", 98);
    S2.cour.setScore("英语", 88);
    S2.cour.setScore("数学", 68);
    S1.cour.Print();
    S2.cour.Print();
}
int main()
{
    TestStudent();
    //system("pause"); 
  return 0;
}
//C++学习笔记_03类的继承
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
class CPerson
{
private:
    char name[16];
protected:        //对象不能直接访问,但是允许子类中的成员函数进行访问
    unsigned int age;
public:
    CPerson();
    CPerson(char szName[], unsigned int uiAge);
    ~CPerson();
    void Print();
};
CPerson::CPerson(){}
CPerson::CPerson(char szName[], unsigned int uiAge){
    cout << "构造 CPerson(char szName[], unsigned int uiAge)" << endl;
    strcpy(name,szName);
    age = uiAge;
}
CPerson::~CPerson(){}
void CPerson::Print(){
    cout << "姓名  :" << name << endl;
    cout << "年龄  :" << age << endl;
}
//CWorker 继承 CPerson: 也就是说,不需要重写 某些函数或者变量
//无论什么继承方式,父类的 private 成员,子类不允许访问
//对于父类的 public 成员,和 protected成员:
//public 表示公有继承方式:父类的成员,访问权限不变              (不降权)
//private 表示私有继承:   父类的成员,相当于子类的 私有成员     (降权)
//protected 表示保护继承:  父类的成员,相当于子类的 protected成员(降权)
class CWorker:public CPerson
{
private:
    char company[20]; //公司
public:
    CWorker(char szName[], unsigned int uiAge, char szCompany[]);
    ~CWorker();
    void Print();
    unsigned int GetAge(){//不允许调用 父类 的私有成员
        return age; //如果我们想调用的话,可以把 age 声明成 protected 成员  
    }
};
CWorker::CWorker(char szName[], unsigned int uiAge, char szCompany[]):CPerson(szName, uiAge){
    //CPerson::CPerson(szName, uiAge); //不能这样写,因为这样写相当于定义一个新 CPerson 对象
    //在外部调用父类构造函数构造对象,
    //如果在函数里面调用,则会生成一个新的CPerson对象(匿名对象)
    strcpy(company,szCompany);
}
CWorker::~CWorker(){}
void CWorker::Print(){
    CPerson::Print();  //指定调用父类的Print函数,
        //不加CPerson:: 则会有限调用自己定义的 Print() --> 无限递归
    cout << "公司  :" << company << endl;
}
void TestInherit()
{
    CWorker W1("小王", 30, "腾信");
    W1.Print(); //事实上就调用了父类的 Print()
    //并不是说父类CPerson 的所有函数,CWorker 都能调用
    //CWorker 不能调用 CPerson 的 private 中的变量和函数
    //W1.age = 40;// 不允许调用父类的私有成员
    //protected:受保护的成员,子类中的函数,可以直接调用。
    //如果子类也定义了 Print() 函数(同名,而且入参一样),那么,会隐藏掉父类的函数。
}
int main()
{
    TestInherit();
    //system("pause"); 
  return 0;
}
//C++学习笔记_03类的继承  多重继承 
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
#define C_MAX 10
class CPerson
{
private:
    char name[16];
protected:        //对象不能直接访问,但是允许子类中的成员函数进行访问
    unsigned int age;
public:
    CPerson();
    CPerson(char szName[], unsigned int uiAge);
    ~CPerson();
    void Print();
};
CPerson::CPerson(){}
CPerson::CPerson(char szName[], unsigned int uiAge){
    cout << "构造 CPerson(char szName[], unsigned int uiAge)" << endl;
    strcpy(name,szName);
    age = uiAge;
}
CPerson::~CPerson(){}
void CPerson::Print(){
    cout << "姓名  :" << name << endl;
    cout << "年龄  :" << age << endl;
}
//CWorker 继承 CPerson: 也就是说,不需要重写 某些函数或者变量
//无论什么继承方式,父类的 private 成员,子类不允许访问
//对于父类的 public 成员,和 protected成员:
//public 表示公有继承方式:父类的成员,访问权限不变              (不降权)
//private 表示私有继承:   父类的成员,相当于子类的 私有成员     (降权)
//protected 表示保护继承:  父类的成员,相当于子类的 protected成员(降权)
class CWorker:public CPerson
{
private:
    char company[20]; //公司
public:
    CWorker(char szName[], unsigned int uiAge, char szCompany[]);
    ~CWorker();
    void Print();
    unsigned int GetAge(){//不允许调用 父类 的私有成员
        return age; //如果我们想调用的话,可以把 age 声明成 protected 成员  
    }
};
CWorker::CWorker(char szName[], unsigned int uiAge, char szCompany[]):CPerson(szName, uiAge){
    //CPerson::CPerson(szName, uiAge); //不能这样写,因为这样写相当于定义一个新 CPerson 对象
    //在外部调用父类构造函数构造对象,
    //如果在函数里面调用,则会生成一个新的CPerson对象(匿名对象)
    strcpy(company,szCompany);
}
CWorker::~CWorker(){}
void CWorker::Print(){
    CPerson::Print();  //指定调用父类的Print函数,
        //不加CPerson:: 则会有限调用自己定义的 Print() --> 无限递归
    cout << "公司  :" << company << endl;
}
class CFarmer:public CPerson
{
private:
    char address[20]; //农民的土地位置
public:
    CFarmer(char szName[], unsigned int uiAge, char szAddr[]);
    ~CFarmer(){}
    void Print();    
};
CFarmer::CFarmer(char szName[], unsigned int uiAge, char szAddr[]) :CPerson(szName, uiAge){
    strcpy(address,szAddr);
}
void CFarmer:: Print(){
    CPerson::Print();
    cout << "土地  :" << address << endl;
}
void TestFarmer()
{
    CFarmer Fa("小张",36,"中国大连");
    Fa.Print(); 
}
class CBase
{
public:
    CBase();
    ~CBase();
    void Print();
};
CBase::CBase() {cout << "构造 CBase..." << endl;}
CBase::~CBase() {cout << "析构 CBase..." << endl;}
void CBase::Print() {cout << "I am CBase..." << endl;}
class CSuper
{
public:
    CSuper();
    ~CSuper();
    void Print();
};
CSuper::CSuper() {cout << "构造 CSuper..." << endl;}
CSuper::~CSuper() {cout << "析构 CSuper..." << endl;}
void CSuper::Print() {cout << "I am CSuper..." << endl;}
//比如说,我们定义学生类 CStudent,和 中国人类 CChinese
//再定义中国的学生类  CChStudent 可以同时继承 CStudent 和 CChinese
//这就是多重继承
//多重继承:一个类,继承两个父类
class Derived :public CBase, public CSuper
{
public:
    Derived();
    ~Derived();
};
Derived::Derived() {cout << "构造 Derived..." << endl;}
Derived::~Derived() {cout << "析构 Derived..." << endl;}
void TestBase()//测试多重继承的运行顺序
{
    Derived  D1;
}
int main()
{
    TestFarmer();
    TestBase();
    //system("pause"); 
  return 0;
}


相关文章
|
1天前
|
编译器 C++
|
1天前
|
编译器 C语言 C++
|
1天前
|
编译器 C++
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
【C++】详解初始化列表,隐式类型转化,类静态成员,友元
|
4天前
|
存储 编译器 C++
【C++】类和对象④(再谈构造函数:初始化列表,隐式类型转换,缺省值
C++中的隐式类型转换在变量赋值和函数调用中常见,如`double`转`int`。取引用时,须用`const`以防修改临时变量,如`const int& b = a;`。类可以有隐式单参构造,使`A aa2 = 1;`合法,但`explicit`关键字可阻止这种转换。C++11起,成员变量可设默认值,如`int _b1 = 1;`。博客探讨构造函数、初始化列表及编译器优化,关注更多C++特性。
|
4天前
|
编译器 C++
【C++】类和对象④(类的默认成员函数:取地址及const取地址重载 )
本文探讨了C++中类的成员函数,特别是取地址及const取地址操作符重载,通常无需重载,但展示了如何自定义以适应特定需求。接着讨论了构造函数的重要性,尤其是使用初始化列表来高效地初始化类的成员,包括对象成员、引用和const成员。初始化列表确保在对象创建时正确赋值,并遵循特定的执行顺序。
|
4天前
|
C语言 C++
【C++】日期类Date(详解)③
该文介绍了C++中直接相减法计算两个日期之间差值的方法,包括确定max和min、按年计算天数、日期矫正及计算差值。同时,文章讲解了const成员函数,用于不修改类成员的函数,并给出了`GetMonthDay`和`CheckDate`的const版本。此外,讨论了流插入和流提取的重载,需在类外部定义以符合内置类型输入输出习惯,并介绍了友元机制,允许非成员函数访问类的私有成员。全文旨在深化对运算符重载、const成员和流操作的理解。
|
4天前
|
C++
【C++】日期类Date(详解)②
- `-=`通过复用`+=`实现,`Date operator-(int day)`则通过创建副本并调用`-=`。 - 前置`++`和后置`++`同样使用重载,类似地,前置`--`和后置`--`也复用了`+=`和`-=1`。 - 比较运算符重载如`&gt;`, `==`, `&lt;`, `&lt;=`, `!=`,通常只需实现两个,其他可通过复合逻辑得出。 - `Date`减`Date`返回天数,通过迭代较小日期直到与较大日期相等,记录步数和符号。 ``` 这是236个字符的摘要,符合240字符以内的要求,涵盖了日期类中运算符重载的主要实现。
|
4天前
|
定位技术 C语言 C++
C++】日期类Date(详解)①
这篇教程讲解了如何使用C++实现一个日期类`Date`,涵盖操作符重载、拷贝构造、赋值运算符及友元函数。类包含年、月、日私有成员,提供合法性检查、获取某月天数、日期加减运算、比较运算符等功能。示例代码包括`GetMonthDay`、`CheckDate`、构造函数、拷贝构造函数、赋值运算符和相关运算符重载的实现。
|
4天前
|
编译器 C++
【C++】类和对象③(类的默认成员函数:赋值运算符重载)
在C++中,运算符重载允许为用户定义的类型扩展运算符功能,但不能创建新运算符如`operator@`。重载的运算符必须至少有一个类类型参数,且不能改变内置类型运算符的含义。`.*::sizeof?`不可重载。赋值运算符`=`通常作为成员函数重载,确保封装性,如`Date`类的`operator==`。赋值运算符应返回引用并检查自我赋值。当未显式重载时,编译器提供默认实现,但这可能不足以处理资源管理。拷贝构造和赋值运算符在对象复制中有不同用途,需根据类需求定制实现。正确实现它们对避免数据错误和内存问题至关重要。接下来将探讨更多操作符重载和默认成员函数。
|
4天前
|
存储 编译器 C++
【C++】类和对象③(类的默认成员函数:拷贝构造函数)
本文探讨了C++中拷贝构造函数和赋值运算符重载的重要性。拷贝构造函数用于创建与已有对象相同的新对象,尤其在类涉及资源管理时需谨慎处理,以防止浅拷贝导致的问题。默认拷贝构造函数进行字节级复制,可能导致资源重复释放。例子展示了未正确实现拷贝构造函数时可能导致的无限递归。此外,文章提到了拷贝构造函数的常见应用场景,如函数参数、返回值和对象初始化,并指出类对象在赋值或作为函数参数时会隐式调用拷贝构造。