C++Primer第五版 8.2.2节练习

时间:2023-02-13 20:04:13

练习8.7:修改上一节的书店程序,将结果保存到一个文件中。将输出文件名作为第二个参数传递给main函数。

练习8.8:修改上一题的程序,将结果追加到给定的文件末尾。对同一个输出文件,运行程序至少两次,检验数据是否得以保留。

答:见 练习8.7.cpp 和 练习8.8.cpp

练习8.7

/*
*练习8.7
*日期:2015/7/23
*问题描述:练习8.7:修改上一节的书店程序,将结果保存到一个文件中。将输出文件名作为第二个参数传递给main函数。
*说明:结果保存到一个文件中
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/



/*
*练习8.6
*日期:2015/7/23
*问题描述:练习8.6:重写7.1.1节书店程序(第229页),从一个文件中读取交易记录。将文件名作为一个参数传递给main(参见6.2.5节,第196页)。
*说明:在练习7.21的基础上改进,新建一个文件sales_data.txt,其他操作与原理几乎一样。
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/



/*
*练习7.21
*日期:2015/7/6
*问题描述:练习7.21:修改你的Sales_data类使其隐藏实现的细节。你之前编写的关于Sales_data操作的程序应该继续使用,借助类的新定义重新编译该程序,确保其工作正常。
*功能;使用访问说明符
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/


#include <iostream>
#include <string>
#include <fstream>
#include <sstream>
using namespace std;

class Sales_data{
public:
friend istream &read(istream &, Sales_data &);
//一个友元函数read,便于Sales_data类直接访问
friend ostream &print(ostream &os, const Sales_data &item);
Sales_data() = default;//默认构造函数
Sales_data(const std::string &s) : bookNo(s){ }
Sales_data(const std::string &s, unsigned n, double p) : bookNo(s), units_sold(n), revenue(p*n){ }
Sales_data(std::istream &is){
read(is,*this);
}


string isbn() const {return bookNo;}
Sales_data& combine(const Sales_data&);
double avg_price() const;

private:
string bookNo;
unsigned units_sold = 0;
double revenue = 0.0;
};

Sales_data add(const Sales_data&, const Sales_data&);
std::ostream &print(std::ostream&, const Sales_data&);
std::istream &read(std::istream&, Sales_data&);

double Sales_data::avg_price() const{
if (units_sold)
return revenue/units_sold;
else
return 0;
}

Sales_data& Sales_data::combine(const Sales_data &rhs)
{
units_sold += rhs.units_sold;
revenue += rhs.revenue;
return *this;
}

istream &read(istream &is, Sales_data &item)
{
double price = 0;
is >> item.bookNo >> item.units_sold >> price;
item.revenue = price * item.units_sold;
return is;
}

ostream &print(ostream &os, const Sales_data &item){
os << item.isbn() << " " << item.units_sold << " "
<< item.revenue << " " << item.avg_price();
return os;
}

Sales_data add(const Sales_data &lhs, const Sales_data &rhs){
Sales_data sum = lhs;
sum.combine(rhs);
return sum;
}

/*
Sales_data::Sales_data(std::istream &is)
{
read(is,*this);
}
*/


int main()
{
ifstream in("sales_data.txt");
//改动地方,新增一个Sales_data.txt
ofstream out("sales_data_save.txt");
//改动地方,新增一个写文件sales_data_save.txt
Sales_data total;
if(read(in, total)){ //read(cin,total); 变为read(in,total);
Sales_data trans;
while(read(in,trans)){ //read(cin, total); 变为read(in,total);
if(total.isbn() == trans.isbn())
total.combine(trans);
else
{
print(cout, total) << endl;
print(out,total) << endl;
total = trans;
}
}
print(cout, total) << endl;
print(out,total) << endl;
} else{
cerr << "No data?!" << endl;
}
}

练习8.8

/*
*练习8.8
*日期:2015/7/23
*问题描述:练习8.8:修改上一题的程序,将结果追加到给定的文件末尾。对同一个输出文件,运行程序至少两次,检验数据是否得以保留。
*说明:注意文件模式
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/


/*
*练习8.7
*日期:2015/7/23
*问题描述:练习8.7:修改上一节的书店程序,将结果保存到一个文件中。将输出文件名作为第二个参数传递给main函数。
*说明:结果保存到一个文件中
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/



/*
*练习8.6
*日期:2015/7/23
*问题描述:练习8.6:重写7.1.1节书店程序(第229页),从一个文件中读取交易记录。将文件名作为一个参数传递给main(参见6.2.5节,第196页)。
*说明:在练习7.21的基础上改进,新建一个文件sales_data.txt,其他操作与原理几乎一样。
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/



/*
*练习7.21
*日期:2015/7/6
*问题描述:练习7.21:修改你的Sales_data类使其隐藏实现的细节。你之前编写的关于Sales_data操作的程序应该继续使用,借助类的新定义重新编译该程序,确保其工作正常。
*功能;使用访问说明符
*作者:Nick Feng
*邮箱:nickgreen23@163.com
*/


#include <iostream>
#include <string>
#include <fstream>
#include <sstream>
using namespace std;

class Sales_data{
public:
friend istream &read(istream &, Sales_data &);//一个友元函数read,便于Sales_data类直接访问
friend ostream &print(ostream &os, const Sales_data &item);
Sales_data() = default;//默认构造函数
Sales_data(const std::string &s) : bookNo(s){ }
Sales_data(const std::string &s, unsigned n, double p) : bookNo(s), units_sold(n), revenue(p*n){ }
Sales_data(std::istream &is){
read(is,*this);
}


string isbn() const {return bookNo;}
Sales_data& combine(const Sales_data&);
double avg_price() const;

private:
string bookNo;
unsigned units_sold = 0;
double revenue = 0.0;
};

Sales_data add(const Sales_data&, const Sales_data&);
std::ostream &print(std::ostream&, const Sales_data&);
std::istream &read(std::istream&, Sales_data&);

double Sales_data::avg_price() const{
if (units_sold)
return revenue/units_sold;
else
return 0;
}

Sales_data& Sales_data::combine(const Sales_data &rhs)
{
units_sold += rhs.units_sold;
revenue += rhs.revenue;
return *this;
}

istream &read(istream &is, Sales_data &item)
{
double price = 0;
is >> item.bookNo >> item.units_sold >> price;
item.revenue = price * item.units_sold;
return is;
}

ostream &print(ostream &os, const Sales_data &item){
os << item.isbn() << " " << item.units_sold << " "
<< item.revenue << " " << item.avg_price();
return os;
}

Sales_data add(const Sales_data &lhs, const Sales_data &rhs){
Sales_data sum = lhs;
sum.combine(rhs);
return sum;
}

/*
Sales_data::Sales_data(std::istream &is)
{
read(is,*this);
}
*/


int main()
{
ifstream in("sales_data.txt");//改动地方,新增一个Sales_data.txt
ofstream out("sales_data_save_second.txt",ofstream::app); //改动地方,新增一个写文件sales_data_save_second.txt,模式为ofstream::app,即写操作在上一个记录的末尾进行操作
Sales_data total;
if(read(in, total)){ //read(cin,total); 变为read(in,total);
Sales_data trans;
while(read(in,trans)){ //read(cin, total); 变为read(in,total);
if(total.isbn() == trans.isbn())
total.combine(trans);
else
{
print(cout, total) << endl;
print(out,total) << endl;
total = trans;
}
}
print(cout, total) << endl;
print(out,total) << endl;
} else{
cerr << "No data?!" << endl;
}
}

sales_data.txt
Hero 1 2
Hero 2 3
Hero 2 4
Bear 1 2
Bear 2 3
Girl 5 5
sales_data_save.txt
Hero 5 16 3.2
Bear 3 8 2.66667
Girl 5 25 5
sales_data_save_second.txt
Hero 5 16 3.2
Bear 3 8 2.66667
Girl 5 25 5
Hero 5 16 3.2
Bear 3 8 2.66667
Girl 5 25 5
Hero 5 16 3.2
Bear 3 8 2.66667
Girl 5 25 5
Hero 5 16 3.2

Bear 3 8 2.66667

Girl 5 25 5

Hero 5 16 3.2
Bear 3 8 2.66667
Girl 5 25 5