2018-08-07

```

//9_16.cpp#include "account.h"#include "Array.h"#includeusing namespace std;int main() {Date date(2008, 11, 1);//起始日期Array accounts(0); //创建账户数组,元素个数为0

cout << "(a)add account (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> ";

char type;

int index, day;

double amount, credit, rate, fee;

string id, desc;

Account* account;

cin >> cmd;

switch (cmd) {

case 'a': //增加账户

cin >> type >> id;

if (type == 's') {

cin >> rate;

account = new SavingsAccount(date, id, rate);

} else {

cin >> credit >> rate >> fee;

account = new CreditAccount(date, id, credit, rate, fee);

}

accounts.resize(accounts.getSize() + 1);

accounts[accounts.getSize() - 1] = account;

break;

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 < accounts.getSize(); 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 < accounts.getSize(); i++)

accounts[i]->settle(date);

break;

}

} while (cmd != 'e');

for (int i = 0; i < accounts.getSize(); i++)

delete accounts[i];

return 0;

}

//account.cpp#include "account.h"#include#includeusing namespace std;

double Account::total = 0;

//Account类的实现

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::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) {

if (date.getMonth() == 1) { //每年的一月计算一次利息

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::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.h#ifndef __ACCOUNT_H__#define __ACCOUNT_H__#include "date.h"#include "accumulator.h"#includeclass Account { //账户类

private:

std::string id; //帐号

double balance; //余额

static double total; //所有账户的总金额

protected:

//供派生类调用的构造函数,id为账户

Account(const Date &date, const std::string &id);

//记录一笔帐,date为日期,amount为金额,desc为说明

void record(const Date &date, double amount, const std::string &desc);

//报告错误信息

void error(const std::string &msg) const;

public:

const std::string &getId() const { return id; }

double getBalance() const { return balance; }

static double getTotal() { return total; }

//存入现金,date为日期,amount为金额,desc为款项说明

virtual void deposit(const Date &date, double amount, const std::string &desc) = 0;

//取出现金,date为日期,amount为金额,desc为款项说明

virtual void withdraw(const Date &date, double amount, const std::string &desc) = 0;

//结算(计算利息、年费等),每月结算一次,date为结算日期

virtual void settle(const Date &date) = 0;

//显示账户信息

virtual void show() const;

};

class SavingsAccount : public Account { //储蓄账户类

private:

Accumulator acc; //辅助计算利息的累加器

double rate; //存款的年利率

public:

//构造函数

SavingsAccount(const Date &date, const std::string &id, double rate);

double getRate() const { return rate; }

virtual void deposit(const Date &date, double amount, const std::string &desc);

virtual void withdraw(const Date &date, double amount, const std::string &desc);

virtual void settle(const Date &date);

};

class CreditAccount : public Account { //信用账户类

private:

Accumulator acc; //辅助计算利息的累加器

double credit; //信用额度

double rate; //欠款的日利率

double fee; //信用卡年费

double getDebt() const { //获得欠款额

double balance = getBalance();

return (balance < 0 ? balance : 0);

}

public:

//构造函数

CreditAccount(const Date &date, const std::string &id, double credit, double rate, double fee);

double getCredit() const { return credit; }

double getRate() const { return rate; }

double getFee() const { return fee; }

double getAvailableCredit() const { //获得可用信用

if (getBalance() < 0)

return credit + getBalance();

else

return credit;

}

virtual void deposit(const Date &date, double amount, const std::string &desc);

virtual void withdraw(const Date &date, double amount, const std::string &desc);

virtual void settle(const Date &date);

virtual void show() const;

};

#endif //__ACCOUNT_H__

//accumulator.h

#ifndef __ACCUMULATOR_H__

#define __ACCUMULATOR_H__

#include "date.h"

class Accumulator { //将某个数值按日累加

private:

Date lastDate; //上次变更数值的时期

double value; //数值的当前值

double sum; //数值按日累加之和

public:

//构造函数,date为开始累加的日期,value为初始值

Accumulator(const Date &date, double value)

: lastDate(date), value(value), sum(0) { }

//获得到日期date的累加结果

double getSum(const Date &date) const {

return sum + value * (date - lastDate);

}

//在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;

}

};

#endif //__ACCUMULATOR_H__

//Array.h#ifndef ARRAY_H#define ARRAY_H#include//数组类模板定义templateclass Array {private:T* list;//T类型指针,用于存放动态分配的数组内存首地址int size;//数组大小(元素个数)public:Array(int sz = 50);//构造函数Array(const Array&a);//拷贝构造函数~Array();//析构函数Array& operator = (const Array&rhs); //重载"="使数组对象可以整体赋值T & operator [] (int i);//重载"[]",使Array对象可以起到C++普通数组的作用const T & operator [] (int i) const;//"[]"运算符的const版本operator T * ();//重载到T*类型的转换,使Array对象可以起到C++普通数组的作用operator const T * () const;//到T*类型转换操作符的const版本int getSize() const;//取数组的大小void resize(int sz);//修改数组的大小};//构造函数templateArray::Array(int sz) {assert(sz >= 0);//sz为数组大小(元素个数),应当非负size = sz;// 将元素个数赋值给变量sizelist = new T [size];//动态分配size个T类型的元素空间}//析构函数templateArray::~Array() {delete [] list;}//拷贝构造函数templateArray::Array(const Array&a) {//从对象x取得数组大小,并赋值给当前对象的成员size = a.size;//为对象申请内存并进行出错检查list = new T[size];// 动态分配n个T类型的元素空间//从对象X复制数组元素到本对象 for (int i = 0; i < size; i++)list[i] = a.list[i];}//重载"="运算符,将对象rhs赋值给本对象。实现对象之间的整体赋值templateArray&Array::operator = (const Array& rhs) {if (&rhs != this) {//如果本对象中数组大小与rhs不同,则删除数组原有内存,然后重新分配if (size != rhs.size) {delete [] list;//删除数组原有内存size = rhs.size;//设置本对象的数组大小list = new T[size];//重新分配n个元素的内存}//从对象X复制数组元素到本对象 for (int i = 0; i < size; i++)list[i] = rhs.list[i];}return *this;//返回当前对象的引用}//重载下标运算符,实现与普通数组一样通过下标访问元素,并且具有越界检查功能templateT &Array::operator[] (int n) {assert(n >= 0 && n < size);//检查下标是否越界return list[n];//返回下标为n的数组元素}templateconst T &Array::operator[] (int n) const {assert(n >= 0 && n < size);//检查下标是否越界return list[n];//返回下标为n的数组元素}//重载指针转换运算符,将Array类的对象名转换为T类型的指针,//指向当前对象中的私有数组。//因而可以象使用普通数组首地址一样使用Array类的对象名templateArray::operator T * () {return list;//返回当前对象中私有数组的首地址}templateArray::operator const T * () const {return list;//返回当前对象中私有数组的首地址}//取当前数组的大小templateint Array::getSize() const {return size;}// 将数组大小修改为sztemplatevoid Array::resize(int sz) {

assert(sz >= 0); //检查sz是否非负

if (sz == size) //如果指定的大小与原有大小一样,什么也不做

return;

T* newList = new T [sz]; //申请新的数组内存

int n = (sz < size) ? sz : size; //将sz与size中较小的一个赋值给n

//将原有数组中前n个元素复制到新数组中

for (int i = 0; i < n; i++)

newList[i] = list[i];

delete[] list; //删除原数组

list = newList; // 使list指向新数组

size = sz; //更新size

}

#endif  //ARRAY_H

//date.cpp#include "date.h"#include#includeusing namespace std;

namespace { //namespace使下面的定义只在当前文件中有效

//存储平年中某个月1日之前有多少天,为便于getMaxDay函数的实现,该数组多出一项

const int DAYS_BEFORE_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_BEFORE_MONTH[month - 1] + day;

if (isLeapYear() && month > 2) totalDays++;

}

int Date::getMaxDay() const {

if (isLeapYear() && month == 2)

return 29;

else

return DAYS_BEFORE_MONTH[month]- DAYS_BEFORE_MONTH[month - 1];

}

void Date::show() const {

cout << getYear() << "-" << getMonth() << "-" << getDay();

}

//date.h

#ifndef __DATE_H__

#define __DATE_H__

class Date { //日期类

private:

int year; //年

int month; //月

int day; //日

int totalDays; //该日期是从公元元年1月1日开始的第几天

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;

}

};

#endif //__DATE_H__

```

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,928评论 6 509
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,748评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,282评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,065评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,101评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,855评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,521评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,414评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,931评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,053评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,191评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,873评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,529评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,074评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,188评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,491评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,173评论 2 357

推荐阅读更多精彩内容

  • Android音频系统详解 参考好文: Android 音频系统:从 AudioTrack 到 AudioFlin...
    爱雨520阅读 13,606评论 2 7
  • mean to add the formatted="false" attribute?.[ 46% 47325/...
    ProZoom阅读 2,699评论 0 3
  • websocket是一种通信协议,和http、tcp/ip协议是一样的,建立在浏览器和服务器之间的长链接应用; 首...
    胡袁强先生阅读 516评论 0 0
  • 2017女王圣诞演讲 嗯,没错,女王少见地摘下了她滴眼镜 关于女王大大纯正发音就然全球皆知就不吧啦吧啦了,圣诞节已...
    Schneee阅读 252评论 0 0
  • 昨天发现公司石头路上有很多地木耳,和倩倩确认后,今天早上跑去摘了一堆,嘿嘿。。。摘起来好过瘾,一下两下,停不下来,...
    无尽夏小柒阅读 160评论 0 0