实验五 继承和多态

任务3:

#ifndef PETS_HPP
#define PETS_HPP

#include <string>
#include <iostream>

class MachinePets {
protected:
    std::string nickname;  

public:
    MachinePets(const std::string& s) : nickname(s) {}

    virtual std::string talk() = 0;

    std::string get_nickname() const {
        return nickname;
    }

    virtual ~MachinePets() {}
};

class PetCats : public MachinePets {
public:
    PetCats(const std::string& s) : MachinePets(s) {}

    std::string talk() override {
        return "喵喵";
    }
};

class PetDogs : public MachinePets {
public:
    PetDogs(const std::string& s) : MachinePets(s) {}

    std::string talk() override {
        return "汪汪";
    }
};

#endif 
pets.hpp
#include <iostream>
#include <vector>
#include "pets.hpp"

void test() {
    using namespace std;

    vector<MachinePets *> pets;

    pets.push_back(new PetCats("miku"));
    pets.push_back(new PetDogs("da huang"));

    for(auto &ptr: pets)
        cout <<ptr->get_nickname() << " says " << ptr->talk() << endl;
}

int main() {
    test();
}
task3.cpp

运行结果:

 

 

 

 

任务4:

#ifndef FILM_HPP
#define FILM_HPP

#include <iostream>
#include <string>

class Film {
private:
    std::string title;        
    std::string director;     
    std::string country;      
    int year;                 

public:
    Film() : title(""), director(""), country(""), year(0) {}

    Film(const std::string& t, const std::string& d, const std::string& c, int y)
        : title(t), director(d), country(c), year(y) {}

    int get_year() const {
        return year;
    }

    friend std::istream& operator>>(std::istream& is, Film& f) {
        is.ignore(std::numeric_limits<std::streamsize>::max(), '\n');
        
        std::cout << "请输入片名: ";
        std::getline(is, f.title); 
        
        std::cout << "请输入导演: ";
        std::getline(is, f.director);
        
        std::cout << "请输入制片国家/地区: ";
        std::getline(is, f.country);
        
        std::cout << "请输入上映年份: ";
        is >> f.year;

        return is;
    }

    friend std::ostream& operator<<(std::ostream& os, const Film& f) {
        os << "片名: " << f.title << std::endl;
        os << "导演: " << f.director << std::endl;
        os << "制片国家/地区: " << f.country << std::endl;
        os << "上映年份: " << f.year << std::endl;
        return os;
    }
};

bool compare_by_year(const Film& f1, const Film& f2) {
    return f1.get_year() < f2.get_year();
}

#endif 
film.hpp
#include "film.hpp"
#include <iostream>
#include <string>
#include <vector>
#include <algorithm>

void test() {
    using namespace std;
    
    int n;
    cout << "输入电影数目: ";
    cin >> n;

    cout << "录入" << n << "部影片信息" << endl;
    vector<Film> film_lst;
    for(int i = 0; i < n; ++i) {
        Film f;
        cout << string(20, '-') << "" << i+1 << "部影片录入" << string(20, '-') << endl;
        cin >> f;
        film_lst.push_back(f);
    }

    // 按发行年份升序排序
    sort(film_lst.begin(), film_lst.end(), compare_by_year);

    cout << string(20, '=') + "电影信息(按发行年份)" +  string(20, '=')<< endl;
    for(auto &f: film_lst)
        cout << f << endl;
}

int main() {
    test();
}
task4.cpp

运行结果:

 

 

 

 

 

任务5:

#ifndef COMPLEX_HPP
#define COMPLEX_HPP

#include <iostream>

template <typename T>
class Complex {
private:
    T real;  
    T imag;  

public:
    Complex() : real(0), imag(0) {}
    Complex(T r, T i) : real(r), imag(i) {}
    T get_real() const { return real; }
    T get_imag() const { return imag; }
    Complex& operator+=(const Complex& other) {
        this->real += other.real;
        this->imag += other.imag;
        return *this;
    }
    Complex operator+(const Complex& other) const {
        return Complex(this->real + other.real, this->imag + other.imag);
    }
    bool operator==(const Complex& other) const {
        return this->real == other.real && this->imag == other.imag;
    }
    friend std::istream& operator>>(std::istream& is, Complex& c) {
        is >> c.real >> c.imag;
        return is;
    }
    friend std::ostream& operator<<(std::ostream& os, const Complex& c) {
        os << c.real << " + " << c.imag << "i";
        return os;
    }
};

#endif 
Complex.hpp
#include "Complex.hpp"
#include <iostream>

using std::cin;
using std::cout;
using std::endl;
using std::boolalpha;

void test1() {
    Complex<int> c1(2, -5), c2(c1);

    cout << "c1 = " << c1 << endl;
    cout << "c2 = " << c2 << endl;
    cout << "c1 + c2 = " << c1 + c2 << endl;
    
    c1 += c2;
    cout << "c1 = " << c1 << endl;
    cout << boolalpha << (c1 == c2) << endl;
}

void test2() {
    Complex<double> c1, c2;
    cout << "Enter c1 and c2: ";
    cin >> c1 >> c2;
    cout << "c1 = " << c1 << endl;
    cout << "c2 = " << c2 << endl;

    cout << "c1.real = " << c1.get_real() << endl;
    cout << "c1.imag = " << c1.get_imag() << endl;
}

int main() {
    cout << "自定义类模板Complex测试1: " << endl;
    test1();

    cout << endl;

    cout << "自定义类模板Complex测试2: " << endl;
    test2();
}
task5.cpp

运行结果:

 

 

 

 

任务6:

#include "date.h"
#include <iostream>
#include <cstdlib>

using namespace std;

namespace {
    const int DAYS_BEFIRE_MONTH[] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };
}

Date::Date(int year, int month, int day) : year(year), month(month), day(day) {
    if (day <= 0 || day > getMaxDay()) {
        cout << "Invalid date: ";
        show();
        cout << endl;
        exit(1);
    }
    int years = year - 1;
    totalDays = years * 365 + years / 4 - years / 100 + years / 400 + DAYS_BEFIRE_MONTH[month - 1] + day;
    if (isLeapYear() && month > 2) totalDays++;
}

int Date::getMaxDay() const {
    if (isLeapYear() && month == 2)
        return 29;
    else
        return DAYS_BEFIRE_MONTH[month] - DAYS_BEFIRE_MONTH[month - 1];
}

void Date::show() const {
    cout << getYear() << "-" << getMonth() << "-" << getDay();
}
account.h
#ifndef __ACCUMULATOR_H__
#define __ACCUMULATOR_H__
#include"date.h"
class Accumulator {//将某a个数值按日累加
private:
    Date lastDate;//上次变更数值的时期
    double value;//数值的当前值
    double sum;//数值按日累加之和
public:
    double getSum(const Date &date)const {
        return sum + value * (date - lastDate);
    }
    Accumulator(const Date &date,double value):lastDate(date),value(value),sum(0){}//构造函数,date为开始累加的日期,value为初始值
    void change(const Date& date, double value) {
        sum = getSum(date);
        lastDate = date; this->value = value;//在date将数值改为value
    }
    void reset(const Date& date, double value) {
        lastDate = date; this->value = value; sum = 0;//初始化,将日期变为date,数值变为value,累加器清零
    }
};
#endif
accumulator.h
#pragma once

class Date {
    private:
        int year;
        int month;
        int day;
        int totalDays;
    public:
        Date(int year, int month, int day);
        int getYear() const {
            return year;
        }
        int getMonth() const {
            return month;
        }
        int getDay() const {
            return day;
        }
        int getMaxDay() const;
        bool isLeapYear() const {
            return year % 4 == 0 && year % 100 != 0 || year % 400 == 0;
        }
        void show() const;
        int operator-(const Date& date) const {
            return totalDays - date.totalDays;
        }
};
date.h
#include "account.h"
#include <cmath>
#include <iostream>

using namespace std;

double Account::total = 0;

Account::Account(const Date& date, const string& id) : id(id), balance(0) {
    date.show();
    cout << "\t#" << id << " created" << endl;
}

void Account::record(const Date& date, double amount, const string& desc) {
    amount = floor(amount * 100 + 0.5) / 100;
    balance += amount;
    total += amount;
    date.show();
    cout << "\t#" << id << "\t" << amount << "\t" << balance << "\t" << desc << endl;
}

void Account::show() const {
    cout << id << "\tBalance:" << balance;
}

void Account::error(const string& msg) const {
    cout << "Error(#" << id << "):" << msg << endl;
}

SavingsAccount::SavingsAccount(const Date& date, const string& id, double rate) 
    : Account(date, id), rate(rate), acc(date, 0) {}

void SavingsAccount::deposit(const Date& date, double amount, const string& desc) {
    record(date, amount, desc);
    acc.change(date, getBalance());
}

void SavingsAccount::withdraw(const Date& date, double amount, const string& desc) {
    if (amount > getBalance()) {
        error("not enough money");
    } else {
        record(date, -amount, desc);
        acc.change(date, getBalance());
    }
}

void SavingsAccount::settle(const Date& date) {
    double interest = acc.getSum(date) * rate / (date - Date(date.getYear() - 1, 1, 1));
    if (interest != 0) record(date, interest, "interest");
    acc.reset(date, getBalance());
}

CreditAccount::CreditAccount(const Date& date, const string& id, double credit, double rate, double fee) 
    : Account(date, id), credit(credit), rate(rate), fee(fee), acc(date, 0) {}

void CreditAccount::deposit(const Date& date, double amount, const string& desc) {
    record(date, amount, desc);
    acc.change(date, getDebt());
}

void CreditAccount::withdraw(const Date& date, double amount, const string& desc) {
    if (amount - getBalance() > credit) {
        error("not enough credit");
    } else {
        record(date, -amount, desc);
        acc.change(date, getDebt());
    }
}

void CreditAccount::settle(const Date& date) {
    double interest = acc.getSum(date) * rate;
    if (interest != 0) record(date, interest, "interest");
    if (date.getMonth() == 1) record(date, -fee, "annual fee");
    acc.reset(date, getDebt());
}

void CreditAccount::show() const {
    Account::show();
    cout << "\tAvailable credit:" << getAvailableCredit();
}
account.cpp
#include "date.h"
#include <iostream>
#include <cstdlib>

using namespace std;

namespace {
    const int DAYS_BEFIRE_MONTH[] = { 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };
}

Date::Date(int year, int month, int day) : year(year), month(month), day(day) {
    if (day <= 0 || day > getMaxDay()) {
        cout << "Invalid date: ";
        show();
        cout << endl;
        exit(1);
    }
    int years = year - 1;
    totalDays = years * 365 + years / 4 - years / 100 + years / 400 + DAYS_BEFIRE_MONTH[month - 1] + day;
    if (isLeapYear() && month > 2) totalDays++;
}

int Date::getMaxDay() const {
    if (isLeapYear() && month == 2)
        return 29;
    else
        return DAYS_BEFIRE_MONTH[month] - DAYS_BEFIRE_MONTH[month - 1];
}

void Date::show() const {
    cout << getYear() << "-" << getMonth() << "-" << getDay();
}
date.cpp
#include"account.h"
#include<iostream>
using namespace std;
int main() {
    Date date(2008, 11, 1);//起始日期
    //建立几个账户
    SavingsAccount sa1(date, "S3755217", 0.015);
    SavingsAccount sa2(date, "02342342", 0.015);
    CreditAccount ca(date, "C5392394", 10000, 0.0005, 50);
    Account* accounts[] = { &sa1,&sa2,&ca };
    const int n = sizeof(accounts) / sizeof(Account*);//账户总数 
    cout << "(d)deposit (w)withdraw (s)show (c)change day (n)next month (e)exit" << endl;
    char cmd;
    do {
        //显示日期和总金额
        date.show();
        cout << "\tTotal:" << Account::getTotal() << "\tcommand>";
        int index, day;
        double amount;
        string desc;
        cin >> cmd;
        switch (cmd) {
        case 'd'://存入现金
            cin >> index >> amount;
            getline(cin, desc);
            accounts[index]->deposit(date, amount, desc);
            break;
        case 'w'://取出现金
            cin >> index >> amount;
            getline(cin, desc);
            accounts[index]->withdraw(date, amount, desc);
            break;
        case 's'://查询各账户信息
            for (int i = 0; i < n; i++) {
                cout << "[" << i << "]";
                accounts[i]->show();
                cout << endl;
            }
            break;
        case 'c'://改变日期
            cin >> day;
            if (day < date.getDay())
                cout << "You cannot specify a previous day";
            else if (day > date.getMaxDay())
                cout << "Invalid day";
            else
                date = Date(date.getYear(), date.getMonth(), day);
            break;
        case 'n'://进入下个月
            if (date.getMonth() == 12)
                date = Date(date.getYear() + 1, 1, 1);
            else
                date = Date(date.getYear(), date.getMonth() + 1, 1);
            for (int i = 0; i < n; i++)
                accounts[i]->settle(date);
            break;
        }
    } while (cmd != 'e');
    return 0;
}
task6.cpp

运行结果:

 

posted @ 2024-12-08 23:12  枫羽秋茜  阅读(2)  评论(0编辑  收藏  举报