12.23
这个题要注意c++对于char的几个常用的方法
#include <iostream>
#include<cstring>
#include<string>
#include <memory>
int main()
{
//两个字符串字面值常量
const char *s1 = "Hello ";
const char *s2 = "World";
//计算长度
unsigned len = strlen(s1) + strlen(s2) ;
char *pc = new char[len]();
//依次赋值过去
strcpy(pc,s1);
strcpy(pc+strlen(s1),s2);
std::cout<<pc<<std::endl;
//两个字符串常量
std::string sa = "Hello ";
std::string sb = "World";
unsigned len1 = strlen(sa.c_str()) + strlen(sb.c_str());
char *ps = new char[len1]();
strcpy(ps,(sa+sb).c_str());
std::cout<<ps<<std::endl;
return 0;
}
12.24
这个题要清楚,只要申请了动态数组,动态数组会根据你往里放元素多少自动增长的
#include <iostream>
#include<cstring>
#include<string>
#include <memory>
int main()
{
//12.24
std::string a;
char *p = new char[2];
std::cout<<"请输入一个字符串"<<std::endl;
std::cin>>a;
strcpy(p,a.c_str());
std::cout<<p<<std::endl;
return 0;
}
12.25
delete [] pa;
12.26
#include <iostream>
#include<cstring>
#include<string>
#include<vector>
#include <memory>
#include<algorithm>
using namespace std;
int main()
{
//1,申请内存
int n = 3;
allocator<string> alloc;
auto p = alloc.allocate(n);
//2,从cin读入内存
string str;
cout<<"please input string:"<<endl;
auto q = p;
while(cin>>str&&q!=(p+n)){
alloc.construct(q++,str);
}
//3,显示并销毁内存
//销毁内存分两步,第一步是destroy执行析构函数,第二部是deallocate释放内存空间
while(q!=p){
cout<<*(--q)<<" ";
}
cout<<endl;
const size_t size = q-p;
while(q!=p+size){
alloc.destroy(q++);
}
alloc.deallocate(p,n);
return 0;
}
12.27
这个玩意儿我自己写的一塌糊涂。。前后写了两三遍,最后还比对改错,map
类与方法如下
#include <fstream>
#include <memory>
#include <vector>
#include <string>
#include <sstream>
#include <map>
#include <set>
#include<algorithm>
using namespace std;
class QueryResult;
class TextQuery{
public:
using line_num = std::vector<std::string>::size_type;
TextQuery(std::ifstream &);
QueryResult query(const std::string &s)const;
private:
std::shared_ptr<std::vector<std::string>> file;//指针指向一块vector
std::map<std::string,std::shared_ptr<set<line_num>>> v_line_map;//value是指针啊
};
class QueryResult
{
public:
friend std::ostream& print(std::ostream&, const QueryResult&);
QueryResult(//关键字
std::string s,
//关键字所在行号
std::shared_ptr<std::set<TextQuery::line_num>> p,
//关键字所在行内容
std::shared_ptr<std::vector<std::string>> f) :
sought(s), lines(p), file(f) {}
private:
std::string sought;//关键字
std::shared_ptr<std::set<TextQuery::line_num>> lines;//set记录行号
std::shared_ptr<std::vector<std::string>> file;//vector记录该行内容
};
TextQuery::TextQuery(ifstream& ifs) : file(new vector<string>) {
std::string text;
while(std::getline(ifs,text)){
file->push_back(text);
int n = file->size()-1;
std::istringstream iss(text);
std::string word;
while(iss>>word){
auto &words = v_line_map[word];
if(!words){
words.reset(new std::set<line_num>);
}
words->insert(n);
}
}
}
QueryResult TextQuery::query(const std::string& s) const
{
static shared_ptr<set<line_num>> nodata(new set<line_num>);
auto loc = v_line_map.find(s);
if (loc == v_line_map.end())
return QueryResult(s, nodata, file);
else
return QueryResult(s, loc->second, file);
}
ostream& print(std::ostream& os, const QueryResult& qr){
std::cout<<"please input you want to select word: "<<endl;
os<<qr.sought<<" occurs "<<qr.lines->size()<<(qr.lines->size()>1?"times":"time")<<endl;
for(auto num:*qr.lines){
os << "\t(line " << num + 1 << ") " << *(qr.file->begin() + num) << endl;
}
}
12.28
这个题目理解了文本程序之后相对简单
#include <fstream>
#include <memory>
#include <vector>
#include <string>
#include <sstream>
#include <map>
#include <set>
#include<algorithm>
#include<iostream>
int main()
{
//1,确定使用什么:一个用与存放文件的每一行的vector,一个map用于指示每个单词与单词出现的行数关系
//2,保存好文件和单词与行号的对应关系之后,
//3,在map中寻找给定的关键字,
//4,找到之后用value确定行号
//5,将vector对应行号的内容输出即可)
using l_int = std::vector<std::string>::size_type;
std::string in = "C:\\study\\c++test\\endless.txt";
std::ifstream ifs(in);
std::map<std::string,std::shared_ptr<std::set<l_int>>> imap;
//用vector存储文件
std::vector<std::string> ivec;
std::string str;
while(getline(ifs,str)){
ivec.push_back(str);
int n = ivec.size()-1;
std::istringstream iss(str);
std::string word;
while(iss>>word){
auto &m = imap[word];
if(!m){
m.reset(new std::set<l_int>);
}
m->insert(n);
}
}
std::string sel;
std::cout<<"please input you want to select word: "<<std::endl;
std::cin>>sel;
auto loc= imap.find(sel);
if(loc == imap.end()){
std::cout<<sel<<"is not in file"<<std::endl;
}else{
for(auto num:*loc->second){
std::cout << "\t(line " << num + 1 << ") " << *(ivec.begin() + num) << std::endl;
}
}
return 0;
}
12.29
因为这里循环条件是true,while与dowhile皆可以,但是如果将打破循环的条件放在循环条件中,就要while更好了,先看do while版本
void runQueries(ifstream &infile){
TextQuery tq(infile);
do{
cout<<"enter wprd to look for , or q to quit :";
string s;
if(!(cin>>s)||s == "q") break;
print(cout,tq.query(s))<<endl;
}while(true);
}
再看加入终止循环条件(dowhile)的版本
void runQueries(ifstream &infile){
TextQuery tq(infile);
string s;
do{
cout<<"enter wprd to look for , or q to quit :";
cin>>s;
print(cout,tq.query(s))<<endl;
}while(true&&s!= "q");
}
再看while的版本
void runQueries(ifstream &infile){
TextQuery tq(infile);
string s;
while(true&&s!= "q"){
cout<<"enter wprd to look for , or q to quit :";
cin>>s;
print(cout,tq.query(s))<<endl;
}
}
很明显,dowhile的版本会先将“q”走一遍业务逻辑,但是while版本就会直接退出
最后再来看一个lamda表达式的版本,使用lamda表达式相对简洁但是可读性较差
void runQueries(ifstream &infile){
TextQuery tq(infile);
string s;
while([&](){cout<<"enter wprd to look for , or q to quit :";return cin>>s&&s!="q";}()){
print(cout,tq.query(s))<<endl;
}
}
12.30
与12.27雷同(100%)
类与方法如下
#include <fstream>
#include <memory>
#include <vector>
#include <string>
#include <sstream>
#include <map>
#include <set>
#include<algorithm>
using namespace std;
class QueryResult;
class TextQuery{
public:
using line_num = std::vector<std::string>::size_type;
TextQuery(std::ifstream &);
QueryResult query(const std::string &s)const;
private:
std::shared_ptr<std::vector<std::string>> file;//指针指向一块vector
std::map<std::string,std::shared_ptr<set<line_num>>> v_line_map;//value是指针啊
};
class QueryResult
{
public:
friend std::ostream& print(std::ostream&, const QueryResult&);
QueryResult(//关键字
std::string s,
//关键字所在行号
std::shared_ptr<std::set<TextQuery::line_num>> p,
//关键字所在行内容
std::shared_ptr<std::vector<std::string>> f) :
sought(s), lines(p), file(f) {}
private:
std::string sought;//关键字
std::shared_ptr<std::set<TextQuery::line_num>> lines;//set记录行号
std::shared_ptr<std::vector<std::string>> file;//vector记录该行内容
};
TextQuery::TextQuery(ifstream& ifs) : file(new vector<string>) {
std::string text;
while(std::getline(ifs,text)){
file->push_back(text);
int n = file->size()-1;
std::istringstream iss(text);
std::string word;
while(iss>>word){
auto &words = v_line_map[word];
if(!words){
words.reset(new std::set<line_num>);
}
words->insert(n);
}
}
}
QueryResult TextQuery::query(const std::string& s) const
{
static shared_ptr<set<line_num>> nodata(new set<line_num>);
auto loc = v_line_map.find(s);
if (loc == v_line_map.end())
return QueryResult(s, nodata, file);
else
return QueryResult(s, loc->second, file);
}
ostream& print(std::ostream& os, const QueryResult& qr){
std::cout<<"please input you want to select word: "<<endl;
os<<qr.sought<<" occurs "<<qr.lines->size()<<(qr.lines->size()>1?"times":"time")<<endl;
for(auto num:*qr.lines){
os << "\t(line " << num + 1 << ") " << *(qr.file->begin() + num) << endl;
}
}
runQuery函数执行
#include <iostream>
#include "class.h"
using namespace std;
void runQueries(ifstream &infile){
TextQuery tq(infile);
string s;
while([&](){cout<<"enter wprd to look for , or q to quit :";return cin>>s&&s!="q";}()){
print(cout,tq.query(s))<<endl;
}
}
int main()
{
string in = "C:\\study\\c++test\\endless.txt";
ifstream ifs(in);
runQueries(ifs);
return 0;
}
12.31
用vector代替set行号,区别:
1,set有序 vector无序,行号不会从上到下显示
2,set不支持重复,vector支持重复,若是某一行有两个相同的字母,这个字母在vector就会有重复的行号,在查询时就会出现重复的结果
12.32
用StrBlob 存储文件代替file;
原本的vector成员改成指向StrBlob 的智能指针;
用StrBlobPtr 访问StrBlob 保存的文件
12.33
在QueryResult类的public中加入这三个方法
std::set<TextQuery::line_no>::iterator begin(){
return lines->begin();
}
std::set<TextQuery::line_no>::iterator end(){
return lines->end();
}
std::shared_ptr<std::vector<std::string>> re_file(){
return shared_ptr<std::vector<std::string>>(file);
}
即可
网友评论