练习11.33:实现你自己版本的单词转换程序。
/*
*C++Primer第五版
*练习11.33
*2015/10/14
*问题描述:练习11.33:实现你自己版本的单词转换程序。
*说明:把书上的程序自己再敲了一遍
*作者:Nick Feng
*邮箱:nickgreen23@163。com
*/
#include <iostream>
#include <vector>
#include <map>
#include <fstream>
#include <string>
#include<sstream>
using namespace std;
//建立转换映射
map<string, string> buildMap(ifstream &map_file)
{
map<string, string> trans_map; //保存转换规则
string key; //要转换的单词
string value; //替换后的内容
//读取第一个单词存入key中,行中剩余内容存入value
while (map_file >> key && getline(map_file, value))
if(value.size() > 1) //检查是否有转换规则
trans_map[key] = value.substr(1); //跳过前导空格
else
throw runtime_error("no rule for " + key);
return trans_map;
}
//生成转换文本
const string & transform(const string &s, const map<string,string> &m)
{
//实际的转换工作;此部分是程序的核心
auto map_it = m.find(s);
//如果单词在转换规则map中
if (map_it != m.cend())
return map_it -> second; //使用替换语句
else
return s; //否则返回原string
}
/*单词转换程序*/
void word_transform(ifstream &map_file, ifstream &input)
{
auto trans_map = buildMap(map_file); //保存转换规则
string text;
while (getline(input, text))
{
istringstream stream(text); //读取每个单词
string word;
bool firstword = true; //控制是否打印空格
while (stream >> word){
if (firstword)
firstword = false;
else
cout << " "; //在单词间打印一个空格
//transform 返回它的第一个参数或其他转换之后的形式
cout << transform(word, trans_map); //打印输出
}
cout << endl; //完成一行的转换
}
}
int main()
{
ifstream file1("map_file.txt"); //转换的规则
ifstream file2("data_input.txt"); //输入的文件
word_transform(file1, file2);
return 0;
}
map_file.txt
brb be right back
k okay?
y why
r are
u you
pic picture
thk thanks!
18r later
data_input.txt
where r u
y dont u send me a pic
k thk 18r
练习11.34:如果你将transform函数中的find替换为下标运算符,会发生什么情况?
答:如果替换为下标运算,程序报错,m.find(s);//是找到关键字是s的迭代器
而前一章说过 m[s]实际是要转换变成的string,是一个mapped_type
练习11.35:在buildMap中,如果进行如下改写,会有什么效果?
trans_map[key] = value.substr(1);
改为trans_map.insert({key,value.substr(1)});
答:运行后的效果是一样的
练习11.36:我们的程序并没有检查输入文件的合法性。特别是,它假定转换规则文件中的规则都是有意义的。如果文件中的某一行包含一个关键字,一个空格,然后就结束了,会发生什么?预测程序行为并进行验证,再与你的程序进行比较。
答:throw一个runtime_error 显示 no rule for “关键字”