习题14.2:
Sales_data.h:
#ifndef SALES_DATA_H
#define SALES_DATA_H
#include
#include
#include
using std::string;
using std::cin; using std::cout; using std::endl;
using std::istream; using std::ostream;
using std::ifstream; using std::ofstream;
//为了在类内定义第四个构造函数所进行的声明
class Sales_data;
istream &operator>>(istream &, Sales_data &);
//类
class Sales_data
{
friend istream &operator>>(istream &is, Sales_data &item);
friend ostream &operator<<(ostream &os, const Sales_data &item);
friend Sales_data operator+(const Sales_data &lhs, const Sales_data &rhs);
public:
//公有成员函数
string isbn()const;
Sales_data &combine(const Sales_data &);
//构造函数
explicit Sales_data(const string &s, unsigned int n, double p) :bookIsbn(s), units_sold(n), revenue(p*n){}
Sales_data() :Sales_data("", 0, 0){}//默认构造函数 委托构造函数
Sales_data(const string &s) :Sales_data(s, 0, 0){ }
Sales_data(istream &is) :Sales_data(){ is >> *this; }
//+=运算符重载
Sales_data &operator+=(const Sales_data &);
private:
//成员变量
string bookIsbn;
unsigned units_sold = 0;
double revenue = 0.0;
//私有成员函数
double avg_price()const;
};
//<<、>>、+运算符重载
Sales_data operator+(const Sales_data &, const Sales_data &);
istream &operator>>(istream &, Sales_data &);
ostream &operator<<(ostream &, const Sales_data &);
#endif
Sales_data.cpp:
#include //使用runtime_error
#include "Sales_data.h"
using std::runtime_error;
//类外定义的构造函数
//成员函数
string Sales_data::isbn()const
{
return bookIsbn;
}
Sales_data & Sales_data::combine(const Sales_data &rhs)
{
units_sold += rhs.units_sold;
revenue += rhs.revenue;
return *this;
}
double Sales_data::avg_price()const
{
return units_sold ? revenue / units_sold : 0;
}
//非成员函数
Sales_data &Sales_data::operator+=(const Sales_data &rhs)
{
while (true)
{
try{
if (this->isbn() == rhs.isbn())
{
this->combine(rhs);
return *this;
}
else
throw runtime_error("输入的ISBN不相同,不能相加!");
}
catch (runtime_error err){
cout << err.what() << endl
<< "Oops, try again? Enter Y/N." << endl;
char c;
cin >> c;
if (!cin || tolower(c) == 'n')
break;
}
}
cout << "输入不正确,接下来的结果不予保证!" << endl;
return Sales_data();//返回一个空对象
}
Sales_data operator+(const Sales_data &lhs, const Sales_data &rhs)
{
Sales_data sum = lhs;
while (true)
{
try{
if (lhs.isbn() == rhs.isbn())
{
sum.combine(rhs);
return sum;
}
else
throw runtime_error("输入的ISBN不相同,不能相加!");
}
catch (runtime_error err){
cout << err.what() << endl
<< "Oops, try again? Enter Y/N." << endl;
char c;
cin >> c;
if (!cin || tolower(c) == 'n')
break;
}
}
cout << "输入不正确,接下来的结果不予保证!" << endl;
return sum;//到这里应该只是随意返回一个错误对象了
}
istream &operator>>(istream &is, Sales_data &item)
{
//cout << "请输入ISBN、已售数目、书本单价:" << endl;
//cout << "read() called." << endl;
double price = 0;
is >> item.bookIsbn >> item.units_sold >> price;
item.revenue = item.units_sold*price;
return is;
}
ostream &operator<<(ostream &os, const Sales_data &item)
{
os << "ISBN编号:" << item.isbn() << " 已售出:"
<< item.units_sold << "本 平均价格:" << item.revenue / item.units_sold
<< "元 总收益:" << item.revenue << "元";
return os;
}
main:
#include "Sales_data.h"
int main(){
cout << "this is a test:" << endl;
string read_file = "售卖记录.txt", write_file = "统计结果.txt";
ifstream in(read_file);
ofstream out(write_file,ofstream::app);
if (in && out)
{
cout << "Open file: " + read_file + " & " + write_file << endl;
Sales_data total;
if (in >> total)
{
Sales_data trans;
while (in >> trans)//到最后一条时 此处不是输入数据的地方,cin返回true
{
if (total.isbn() == trans.isbn())//可在else下加入cout测得trans.bookIsbn为空,返回0,进入else
total.combine(trans);
else
{
out << total << endl;//这样写可以保证最后一条数据正常显示
total = trans;
}
}
out << total << endl;
}
else
{
cout << "特么的啥也没有?" << endl;
}
}
else
cout << "Cannot open file:" + read_file + "&" + write_file << endl;
getchar();
getchar();
return 0;
}
String.h:
#ifndef STRING_H
#define STRING_H
#include
#include
#include
#include
using std::cout; using std::endl;
using std::istream; using std::ostream;
#ifndef _MSC_VER
#define NOEXCEPT noexcept
#else
#define NOEXCEPT
#endif
class String
{
friend ostream &operator<<(ostream &, const String &);
public:
//移动构造函数&移动赋值运算符
String(String &&)NOEXCEPT;
String &operator=(String &&)NOEXCEPT;
//构造函数
String()
:String(""){
cout << "默认拷贝构造函数" << endl;
};
String(const char *);
//拷贝构造函数
String(const String &);
//拷贝赋值运算符
String &operator=(const String &rhs);
//析构函数
~String();
std::size_t size(){ return first_free - element; }
std::size_t capacity(){ return cap - element; }
void push_back(const char &c);
private:
std::allocator alloc;
char *element;
char *first_free;
char *cap;
std::pair alloc_n_copy(const char *, const char *);
void range_init(const char *, const char *);
void check_n_alloc();
void free();
void reallocate();
};
//输出运算符
ostream &operator<<(ostream &, const String &);
#endif
#include "String.h"
String::String(String &&rhs) NOEXCEPT
:element(rhs.element), first_free(rhs.first_free), cap(rhs.cap)
{
cout << "移动构造函数" << endl;
rhs.element = rhs.first_free = rhs.cap = nullptr;
}
String &String::operator=(String &&rhs)NOEXCEPT
{
cout << "移动赋值运算符" << endl;
if (this != &rhs)
{
free();
element = rhs.element;
first_free = rhs.first_free;
cap = rhs.cap;
rhs.element = rhs.first_free = rhs.cap = nullptr;
}
return *this;
}
void String::check_n_alloc()
{
//std::cout << "check_n_alloc~" << std::endl;
if (size() == capacity())
reallocate();
}
void String::push_back(const char &c)
{
//std::cout << "push_back~" << std::endl;
check_n_alloc();
alloc.construct(first_free++, c);
}
void String::reallocate()
{
//std::cout << "reallocate~" << std::endl;
auto new_capacity = size() ? 2 * size() : 1;
auto new_data = alloc.allocate(new_capacity);
auto dest = new_data;
auto elem = element;
for (size_t i = 0; i != size(); ++i)
alloc.construct(dest++, std::move(*elem++));
free();
element = new_data;
cap = first_free = element + new_capacity;
}
std::pair String::alloc_n_copy(const char *beg, const char *end)
{
//std::cout << "alloc_n_copy~" << std::endl;
auto new_beg = alloc.allocate(end - beg);
return{ new_beg, std::uninitialized_copy(beg, end, new_beg) };
}
void String::range_init(const char *beg, const char *end)
{
//std::cout << "range_init~" << std::endl;
auto new_data = alloc_n_copy(beg, end);
element = new_data.first;
first_free = new_data.second;
}
String::String(const char *pc_beg)
{
std::cout << "C风格字符串构造函数调用~" << std::endl;
auto pc_end = const_cast(pc_beg);
while (*pc_end)
++pc_end;
range_init(pc_beg, ++pc_end);
}
String::String(const String &rhs)
{
std::cout << "拷贝构造函数调用~" << std::endl;
range_init(rhs.element, rhs.first_free);
}
String &String::operator=(const String &rhs)
{
std::cout << "赋值运算符调用~" << std::endl;
auto data = alloc.allocate(rhs.first_free - rhs.element);
free();
element = rhs.element;
first_free = rhs.first_free;
return *this;
}
void String::free()
{
std::cout << "free~" << std::endl;
if (element)
std::for_each(element, first_free, [this](const char &c){alloc.destroy(&c); });
alloc.deallocate(element, first_free - element);
element = first_free = nullptr;
}
String::~String()
{
free();
}
//overloaded operator functions
ostream &operator<<(ostream &os, const String &s)
{
for (auto beg = s.element; beg != s.first_free; ++beg)
os << *beg;
return os;
}
#include "String.h"
#include
String func()
{
String ret("a");
return ret;
}
int main()
{
//String s = func();
String ss( "hello" );
cout << ss << endl;
getchar();
return 0;
}
//运算符重载
bool operator==(const StrVec &lhs, const StrVec &rhs)
{
return lhs.size() == rhs.size() && std::equal(lhs.begin(), lhs.end(), rhs.begin());
}
bool operator!=(const StrVec &lhs, const StrVec &rhs)
{
return !(lhs == rhs);
}
begin() end()都是定义的有const限定符的成员函数,只是返回数据成员,即那几个指针。
习题14.18:
bool operator<(const StrVec &lhs, const StrVec &rhs)
{
auto lp = lhs.begin(), rp = rhs.begin();
while (lp != lhs.end() && rp != rhs.end())
{
if (*lp < *rp)
return true;
else if (*lp>*rp)
return false;
++lp, ++rp;
}
if (lp == lhs.end() && rp!=rhs.end())
return true;
else
return false;
}
bool operator<=(const StrVec &lhs, const StrVec &rhs)
{
return !(rhs < lhs);
}
bool operator>(const StrVec &lhs, const StrVec &rhs)
{
return rhs < lhs;
}
bool operator>=(const StrVec &lhs, const StrVec &rhs)
{
return !(lhs < rhs);
}
StrVec &StrVec::operator=(std::initializer_list slst)//构造函数已经有了就用啊
{
*this = StrVec(slst);
return *this;
}
char &String::operator[](std::size_t n)
{
return element[n];
}
const char &String::operator[](std::size_t n)const
{
return element[n];
}
#ifndef PRINTPAINT_H
#define PRINTPAINT_H
#include
#include
using std::cin; using std::string; using std::istream;
class GetInput
{
public:
GetInput(std::istream &i = std::cin) :is(i){ }
string operator()()const
{
string s;
getline(is, s);
return is ? s : string();
}
private:
istream &is;//注意这里一定是引用
};
#endif
#include "标头.h"
#include
int main()
{
std::vector givec;
GetInput gi;
for (string temp; !(temp = gi()).empty();
givec.push_back(temp));
for (auto g : givec)
std::cout << g << " ";
std::cout << std::endl;
getchar();
return 0;
}
.h
#ifndef PRINTPAINT_H
#define PRINTPAINT_H
#include
#include
using std::cin; using std::string; using std::istream;
class Equal
{
public:
Equal(const int &ii) :i(ii){ };
bool operator()(const int &i2)const
{
return i == i2;
}
private:
int i;
};
#endif
.cpp
#include "标头.h"
#include
#include
using std::vector;
int main()
{
vector ivec{ 1, 2, 3, 4, 5, 6, 7, 8, 9, 5, 2, 4, 6, 1, 3 };
//Equal eq(5);
//for (auto beg = ivec.begin();
//(beg = std::find_if(beg, ivec.end(), eq)) != ivec.end();
//*beg = 42);
std::replace_if(ivec.begin(), ivec.end(), Equal(5), 42);
for (auto i : ivec)
std::cout << i << " ";
std::cout << std::endl;
getchar();
return 0;
}
#ifndef PRINTPAINT_H
#define PRINTPAINT_H
#include
#include
using std::cin; using std::string; using std::istream;
class Count
{
public:
Count(std::size_t t) :n(t){ };
bool operator()(const string &s)
{
return s.size() == n;
}
private:
std::size_t n;
};
#endif
#include "标头.h"
#include
#include
#include
#include
using std::vector;
int main()
{
std::ifstream file("有标点单词段落测试.txt");
vector svec;
string s_temp;
while (file >> s_temp)
{
string s_org;
std::remove_copy_if(s_temp.begin(), s_temp.end(), std::back_inserter(s_org), ispunct);
svec.push_back(s_org);
std::cout << s_org << std::endl;
}
vector ivec;
for (std::size_t i = 0; i != 10; ++i)
ivec.push_back(std::count_if(svec.begin(), svec.end(), Count(i + 1)));
for (auto i : ivec)
std::cout << i << " ";
std::cout << std::endl;
getchar();
return 0;
}
#ifndef PRINTPAINT_H
#define PRINTPAINT_H
#include
#include
using std::cin; using std::string; using std::istream;
class Count
{
public:
Count(std::size_t low, std::size_t high)
:_low(low), _high(high){ };
bool operator()(const string &s)
{
return s.size() >= _low && s.size() <= _high;
}
private:
std::size_t _low;
std::size_t _high;
};
#endif
#include "标头.h"
#include
#include
#include
#include
using std::vector;
int main()
{
std::ifstream file("有标点单词段落测试.txt");
vector svec;
string s_temp;
while (file >> s_temp)
{
string s_org;
std::remove_copy_if(s_temp.begin(), s_temp.end(), std::back_inserter(s_org), ispunct);
svec.push_back(s_org);
std::cout << s_org << std::endl;
}
vector ivec;
ivec.push_back(std::count_if(svec.begin(), svec.end(), Count(1, 9)));
ivec.push_back(std::count_if(svec.begin(), svec.end(), Count(10, 100)));
for (auto i : ivec)
std::cout << i << " ";
std::cout << std::endl;
getchar();
return 0;
}
int n;
cin >> n;
vector ivec{ 2, 3, 5 };
std::modulus mod;
auto predicate = [&](const int i){return 0 != mod(n, i); };
auto iter = std::find_if(ivec.begin(), ivec.end(), predicate);
if (iter == ivec.end())
std::cout << "true" << std::endl;
else
std::cout << "false" << std::endl;
习题14.44:
#ifndef CALCULATOR_H
#define CALCULATOR_H
#include
#include
#include