C++ Primer TextQuery 完整代码

最近看这本书, 然后查资料整理了一些这个代码, 并添加了注释 练习一些 IO 和map set吧:

#include "TextQuery.h"

using namespace std;

ifstream &open_file(ifstream &in, const string &file);
void print_results(const set<TextQuery::line_no>& locs, const string& sought, const TextQuery &file);
string make_plural(size_t ctr, const string &word, const string &ending);

int main(int argc, char **argv) {
	ifstream infile;
	if (argc < 2 || !open_file(infile, argv[1])) { //通过参数输入文件 这里如果有文件已经通过openfile打开了文件
		cerr << "No input file!" << endl;
		return EXIT_FAILURE;
	}
	TextQuery tq;
	tq.read_file(infile); //调用TextQuery的函数打开文件
	while (true) {
		cout << "enter word to look for,or q to quit" << endl;
		string s;
		cin >> s;
		if (!cin || s == "q")
			break;
		set<TextQuery::line_no> locs = tq.run_query(s); //输入一个单词, 就可以通过这个函数 输出一个含有这个单词的set
		print_results(locs, s, tq);
	}
	return 0;
}
//打开文件的工具函数
ifstream &open_file(ifstream &in, const string &file)
{
	in.close();       // close in case it was already open
	in.clear();      // clear any existing errors
					 // if the open fails, the stream will be in an invalid state
	in.open(file.c_str()); // open the file we were given 这是真正打开文件的地方
	return in; // condition state is good if open succeeded
}
//打印结果的函数
void print_results(const set<TextQuery::line_no>& locs, const string& sought, const TextQuery &file)
{
	typedef set<TextQuery::line_no> line_nums; //遍历set获取输出的结果
	line_nums::size_type size = locs.size();
	cout << "\n" << sought << " occurs " << size << " "
			<< make_plural(size, "time", "s") << endl;
	line_nums::const_iterator it = locs.begin();
	for (; it != locs.end(); ++it) {
		cout << "\t(line" << (*it) + 1 << ")" << file.text_line(*it) << endl;
	}
}
//显示time 还是times
string make_plural(size_t ctr, const string &word, const string &ending)
{
	return (ctr == 1) ? word : word + ending;
}


#include<iostream>
#include<fstream>       //ifstream 的头文件
#include<set>
#include<string>
#include<vector>
#include<map>
#include <stdlib.h>   //EXIT_FALURE 的头文件
#include<sstream>    //istringstream的头文件
#include<utility> //pairs的头文件

using namespace std;


class TextQuery {
public:
	typedef std::vector<std::string>::size_type line_no;
	//读取文件的 ifstream 并进行逐行读取,然后进行逐个单词 和其对应行号的扫描
	void read_file(std::ifstream &is) {
		store_file(is);
		build_map();
	}
	std::set<line_no> run_query(const std::string &) const;
	std::string text_line(line_no) const;
private:
	void store_file(std::ifstream&);
	void build_map();
	std::vector<std::string> lines_of_text; //存储text文件的每一行
	std::map<std::string, std::set<line_no> > word_map;
};



#include "TextQuery.h"
#include <stdexcept>

//扫描文件中的每一行
void TextQuery::store_file(ifstream &is)
{
	string textline;
	while (getline(is, textline))
		lines_of_text.push_back(textline);
}
//扫描每一行的每个单词
void TextQuery::build_map()
{
	for (line_no line_num = 0; line_num != lines_of_text.size(); ++line_num) {
		istringstream line(lines_of_text[line_num]);
		string word;
		while (line >> word) //遍历每个单词 把num 和word添加到 word_map中
			word_map[word].insert(line_num);
	}
}
//给定一个搜索词, 获取存在该词的行号
set<TextQuery::line_no> TextQuery::run_query(const string &query_word) const
{
	map<string, set<line_no> >::const_iterator loc = word_map.find(query_word);
	if (loc == word_map.end())
		return set<line_no>();
	else
		return loc->second;
}
//根据行号, 把行打印出来
string TextQuery::text_line(line_no line) const
{
	if (line < lines_of_text.size())
		return lines_of_text[line];
	throw std::out_of_range("line number out of range");
}





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值