C++学习路线(二十七)

虚基类

菱形继承的重大缺点

cpp 复制代码
#include <iostream>
using namespace std;

class Tel { // 电话类
public:
	Tel() {
		this->number = 0;
	}
protected:
	int number;
};

class FixLine : public Tel { // 固定电话类
};

class MobilePhone : public Tel { // 移动电话类
};
//无线座机
class WirelessLine : public MobilePhone, public FixLine{ // 无线电话类
public:
	void setNumber(int num) {
		this->Tel::number = num;
	}
	int getNumber() {
		return this->Tel::number;
	}


};
int main() {

	return 0;
}

比如说FixLine,继承于Tel MobilePhone也继承于Tel 那么如果调用number 就会报错

这种情况可以使用虚继承和虚基类进行解决。

cpp 复制代码
#include <iostream>
using namespace std;

class Tel { // 电话类
public:
	Tel() {
		this->number = 0;
	}
protected:
	int number;
};

class FixLine : virtual public Tel { // 固定电话类

};

class MobilePhone : virtual public Tel { // 移动电话类

};
//无线座机
class WirelessLine : public MobilePhone, public FixLine{ // 无线电话类
public:
	void setNumber(int num) {
		this->number = num;
	}
	int getNumber() {
		return this->number;
	}


};
int main() {

	return 0;
}

之前的相亲项目的改造

对Girl类和Boy类进行重构:

  1. 抽取出共同的方法和数据成员, 用来构建一个新的基类[父类]
  2. Girl和Boy类修改为从Human类继承,并修改相关代码。

Human.h

#pragma once
#include <string>
class Humans {
public:
    Humans(std::string name, int age) :   name(name), age(age) {}
    int getAge() const;
    std::string getName() const;
    std::string Description() const;
private:
    std::string name;
    int age;
};

Human.cpp

#include "Human.h"
#include "sstream"
Humans::Humans(std::string name, int age)
{
    this->name = name;
    this->age = age;
}
int Humans::getAge() const {
    return age;
}
std::string Humans::getName() const {
    return name;
}
std::string Humans::Description() const {
    std::stringstream ss;
    ss << "Name: " << name << ", Age: " << age;
    return ss.str();
}

boy.h

#pragma once
#include <string>
#include <vector>
#include "../继承/Human.h"
class Girl;

class Boy : public Humans {
public:
	Boy(std::string name, int age, int salary);
	std::string Description() const;
	int getSalary() const;
	bool satisfy(const Girl& other);
	static void inputsBoys(std::vector<Boy>& boys);
private:
	int salary;
};

boy.cpp

#include <sstream>
#include <string>
#include "boy.h"
#include "girl.h"

#define SALARY_FACTOR 0.005

Boy::Boy(std::string name, int age, int salary) : Humans(name, age) , salary(salary) {}
int Boy::getSalary() const {
    return salary;
}
std::string Boy::Description() const {
    std::stringstream ss;
    ss << getName() << "工资" << salary << " 年龄" << getAge();
    return ss.str();
}
void Boy::inputsBoys(std::vector<Boy>& boys) {
    int n = 1;
    std::string name;
    int age;
    int salary;
    while (1) {
        std::cout << "请输入第" << n << "位男孩的信息(输入0结束输入):" << std::endl;
        std::cout << "年龄:" << std::endl;
        std::cin >> age;
        if (age == 0) break;
        std::cout << "姓名:" << std::endl;
        std::cin >> name;
        std::cout << "工资:" << std::endl;
        std::cin >> salary;
        boys.push_back(Boy(name, age, salary));
        n++;
    }
}

bool Boy::satisfy(const Girl& girl) {
    if (girl.getBeauty() >= salary * SALARY_FACTOR) {
        return true;
    }
    else return false;
}

girl.h

#pragma once
#include <iostream>
#include <string>
#include <vector>
#include "../继承/Human.h"
class Boy;

class Girl : public Humans {
public:
	Girl(std::string name, int age, int beauty);
	std::string Description() const;
	int getBeauty() const;
	bool satisfy(const Boy& boy) const;
	static void inputsGirls(std::vector<Girl>& girls);
private:
	int beauty;
};

girl.cpp

#include "girl.h"
#include "boy.h"
#include <string>
#include <sstream>

#define YANZHI_FACTOR 1.1

Girl::Girl(std::string name, int age, int beauty) : Humans(name, age), beauty(beauty) {}
int Girl::getBeauty() const {
    return beauty;
}
std::string Girl::Description() const {
    std::stringstream ss;
    ss << getName() << "颜值" << beauty << " 年龄" << getAge();
    return ss.str();
}
bool Girl::satisfy(const Boy& boy) const {
    if(boy.getSalary() >= beauty * YANZHI_FACTOR) return true;
    return false;
}
void Girl::inputsGirls(std::vector<Girl>& girls) {
    int age;
    std::string name;
    int beauty;
    int n = 1;
    while (1) {
        std::cout << "请输入第" << n << "位女孩的信息(输入0结束输入):" << std::endl;
        std::cout << "年龄:" << std::endl;
        std::cin >> age;
        if (age == 0) break;
        std::cout << "姓名:" << std::endl;
        std::cin >> name;
        std::cout << "颜值:" << std::endl;
        std::cin >> beauty;
        n++;
        girls.push_back(Girl(name, age, beauty));
    }
}

main.cpp不变

常见错误总结

默认访问权限和语法要求

默认访问权限是private

访问权限之后一定要加冒号:

指定为某种访问权限之后, 就一直是这种权限, 除非再次指定为其他权限.

cpp 复制代码
class MyClass{
    void func(); // private
}

类的成员的访问权限, 与继承方式的区别

相同点:都有public, private, protected

不同点:意义完全不同。

小结:

成员的访问权限:

public, 可以通过外部来访问(通过对象直接访问), 类似于C语言结构体中的成员

private, 只能在内部访问(在这个类的成员函数内访问),但是在子类的内部不能直接访问。

protected, 只能在内部访问,而且可以在子类的内部直接访问。

继承方式:

public, 父类成员, 被继承到子类后,访问权限都不变。

private, 父类成员, 被继承到子类后,访问权限都变为private

protected, 父类成员, 被继承到子类后,public权限的成员,降级为protected, 其他不变。

项目扩展-位图算法

问题:

有很多不重复的整数, 其中最大值不超过40亿, 最小值是0.

要求判断某个指定的整数, 是否在这个集合中.

cpp 复制代码
#include <iostream>
using namespace std;

void init(char* data, int len) {
	unsigned int n = len * 8;
	for (int i = 0; i < n; i++) {
		if (i % 3 == 0) {
			cout << i << " ";
			char* p = data + i / 8;
			*p |= (1 << (i % 8));
		}
	}
	cout << endl;
}

bool check(char* data, int value) {
	char* p = data + value / 8;
	bool ret =  *p & (1 << (value % 8));
	return ret;
}

int main() {
	unsigned int n = 400;
	int len = n / 8 + 1;
	char* data = (char*)malloc(len);
	memset(data, 0, len);
	init(data, len);
	while (1) {
		int value;
		cin >> value;
		if (value == -1) break;
		if (check(data, value)) cout << "1" << endl;
		else cout << "0" << endl;
	}
	free(data);
	return 0;
}

项目练习一-练习一

定义一个类Book, 用来描述新书, 具有以下功能:
查看当前价格.

查看当前的书号

定义一个类SellBook, 用来表示促销的书籍, 要求继承自Book

具有以下功能:

  1. 查看当前折扣
  2. 设置当前折扣
  3. 查看当前的促销价

Book.h

cpp 复制代码
#pragma once
#include <iostream>
#include <string>
class Book {
public:
	Book(float price, std::string isbn);
	~Book();
	float getPrice() const;
	std::string getISBN() const;
protected:
	double price;
	std::string isbn;
};

Book.cpp

cpp 复制代码
#include "Book.h"

Book::Book(float price, std::string isbn) {
    this->price = price;
    this->isbn = isbn;
}

Book::~Book() {

}

float Book::getPrice() const {
    return price;
}
std::string Book::getISBN() const {
    return isbn;
}

SellBook.h

cpp 复制代码
#pragma once
#include "Book.h"
class SellBook : public Book {
public:
	SellBook(float price, std::string isbn, float discount);
	~SellBook();
	float getPrice() const;
	float getDiscount() const;
private:
	float discount;
};

SellBook.cpp

cpp 复制代码
#include "SellBook.h"

SellBook::SellBook(float price, std::string isbn , float discount) : Book(price, isbn), discount(discount){}
SellBook::~SellBook(){}

float SellBook::getPrice() const {
	return price * discount / 10.0;
}
float SellBook::getDiscount() const {
	return discount;
}

main.cpp

cpp 复制代码
#include <iostream>
#include "Book.h"
#include "SellBook.h"
using namespace std;

int main()
{
    Book b1(30, "123-123");
    SellBook sb1(30, "456-456", 5);
    cout << "Book price: " << b1.getPrice() << endl;
    cout << "SellBook price: " << sb1.getPrice() << endl;

    return 0;
}
相关推荐
人才程序员19 分钟前
【C++拓展】vs2022使用SQlite3
c语言·开发语言·数据库·c++·qt·ui·sqlite
不能只会打代码33 分钟前
蓝桥杯例题一
算法·蓝桥杯
OKkankan39 分钟前
实现二叉树_堆
c语言·数据结构·c++·算法
Ciderw2 小时前
MySQL为什么使用B+树?B+树和B树的区别
c++·后端·b树·mysql·面试·golang·b+树
yerennuo2 小时前
windows第七章 MFC类CWinApp介绍
c++·windows·mfc
Ronin-Lotus2 小时前
上位机知识篇---ROS2命令行命令&静态链接库&动态链接库
学习·程序人生·机器人·bash
ExRoc2 小时前
蓝桥杯真题 - 填充 - 题解
c++·算法·蓝桥杯
利刃大大2 小时前
【二叉树的深搜】二叉树剪枝
c++·算法·dfs·剪枝
Kasper01213 小时前
认识Django项目模版文件——Django学习日志(二)
学习·django
索然无味io4 小时前
XML外部实体注入--漏洞利用
xml·前端·笔记·学习·web安全·网络安全·php