c++primerテキストクエリーシステム


以下はc++pirmerのテキストクエリーシステムが再プローブしたソースコードで、基本部分は何も修正されていません.コンパイル環境はc-free 5.0、コンパイルされました.ここでいくつかの注意事項を提出します:1.c++11の特性を使用する場合、コンパイル環境に指令-std=c++11を追加する.コードに表示されるshared_ptr、make_ptrはもともとboostライブラリの内容であり,c++標準ライブラリに吸収されており,ヘッダファイルmemoryを参照すればよい.2.15.34の練習をしている間にコンパイル中にLink error undefined reference to「vtable for XXX」というエラーが発生します.これは、虚関数がリンクされている間に関数体がなく、eval関数が定義されていないためです.たとえば次のクラスでは、コンパイルがエラーになります.
class Base
{
virtual void f() = 0;
}
class Derived : public Base
{
void f();
}
の解決策は、クラス外で関数体を定義すればよい.
void Derived::f()
{}

3.3つ目も注意事項ではありません.プログラムは前の練習を前提にしなければなりません.ここではもう完成しました.また、line_を使用していますNoこの属性の場合、事前に宣言しておくと、この行のコードが漏れているようで、定義されていないエラーが発生します.
#include <string>
#include <vector>
#include <iostream>
#include <fstream>
#include <sstream>
#include <set>
#include <map>
#include <memory>
#include <algorithm> 
using namespace std;
string make_plural(size_t ctr,const string& word,const string &end){
    return (ctr==1)?word:word+end;
}
class QueryResult;
class TextQuery{
    public:
        using line_no=vector<string>::size_type;
        TextQuery(ifstream&);
        QueryResult query(const string&) const;
    private:
        shared_ptr<vector<string>> file;
        map<string,shared_ptr<set<line_no>>> wm;

};
TextQuery::TextQuery(ifstream& is):file(new vector<string>){
    string text;
    while(getline(is,text)){
        file->push_back(text);
        int n=file->size()-1;
        istringstream line(text);
        string word;
        while(line>>word){
            auto &lines=wm[word];
            if(!lines){
                lines.reset(new set<line_no>);  
            }
            lines->insert(n);
        }
            }
};
class QueryResult{
    friend ostream& print(ostream&,const QueryResult&);
    public:
        using line_no=TextQuery::line_no;
        using ResultIter=set<line_no>::iterator;
        QueryResult(string s,
                    shared_ptr<set<line_no> > p,
                    shared_ptr<vector<string>> f):
            sought(s),lines(p),file(f){}
        shared_ptr<vector<string>> get_file()const {
            return file;
        }
        ResultIter begin()const {
            return lines->begin();
        }
        ResultIter end() const{
            return lines->end();
        }
    private:
        string sought;
        shared_ptr<set<line_no> > lines;
        shared_ptr<vector<string>> file;
};
QueryResult
TextQuery::query(const string&sought) const{
    static shared_ptr<set<line_no>> nodata(new set<line_no>);
    auto loc=wm.find(sought);
    if(loc==wm.end())
        return QueryResult(sought,nodata,file);
    else
        return QueryResult(sought,loc->second,file);

};
ostream &print(ostream& os,const QueryResult &qr){
    os<<qr.sought<<"occur "<<qr.lines->size()<<" "
        <<make_plural(qr.lines->size(),"times","s")<<endl;
    for(auto num:*qr.lines){
        os<<"\t(line"<<num+1<<")"
            <<*(qr.file->begin()+num)<<endl;
    }
    return os;
}
class Query_base{
    friend class Query;
    protected:
        using line_no=TextQuery::line_no;
        virtual ~Query_base(); 
    private:
        virtual QueryResult eval(const TextQuery&)const =0;
        virtual string rep()const=0;
} ;
Query_base::~Query_base(){

}
class Query{
    friend Query operator~(const Query&);
    friend Query operator|(const Query&,const Query&);
    friend Query operator&(const Query&,const Query&);
    public:
        Query(const string&);
        QueryResult eval(const TextQuery &t)const{
            return q->eval(t);
            cout<<"WordQuery eval"<<endl;
        }
        string rep()const{
            cout<<"Query rep"<<endl;
            return q->rep();

        }
    private:
        Query(shared_ptr<Query_base> query):q(query){

        }
        shared_ptr<Query_base> q;
};
ostream &
operator<<(ostream &os,const Query &query){
    return os<<query.rep();
}
class WordQuery:public Query_base{
    friend class Query;
    WordQuery(const string &s):query_word(s){
            cout<<"WordQuery construct "<<s<<endl;
    }
    QueryResult eval(const TextQuery &t) const{
        cout<<"WordQuery eval"<<endl;
        return t.query(query_word);

    }
    string rep()const{
        cout<<"WordQuery rep"<<endl;
        return query_word;

    }
    string query_word;
};
inline Query::Query(const string &s):q(new WordQuery(s)){
    cout<<"Query construct"<<s<<endl;
}
class NotQuery:public Query_base{
    friend Query operator~(const Query &);
    NotQuery(const Query&q):query(q){
        cout<<"NotQuery construct"<<endl;       
    }
    string rep()const{
        return "~("+query.rep()+")";
    }
    QueryResult eval(const TextQuery&) const;   
    Query query;    
};
inline Query operator~(const Query &operand){
    return shared_ptr<Query_base>(new NotQuery(operand));
}
QueryResult
NotQuery::eval(const TextQuery& text)const{
    auto result=query.eval(text);
    auto ret_lines=make_shared<set<line_no>>();
    auto beg=result.begin(),end=result.end();
    auto sz=result.get_file()->size();
    for(size_t n=0;n!=sz;++n){
        if(beg==end||*beg!=n)
            ret_lines->insert(n);
        else if(beg!=end)
            ++beg;
    }
    return QueryResult(rep(),ret_lines,result.get_file());
}
class Binary_Query:public Query_base{
    protected:
        Binary_Query(const Query &l,const Query &r,string s):
            lhs(l),rhs(r),opSym(s){
                cout<<"Binary_Query construct"<<s<<endl;    
            }
        string rep()const{
            cout<<"Binary_Query rep"<<endl;
            return "("+lhs.rep()+" "
                      +opSym+" "
                      +rhs.rep()+")";

        }

    Query lhs,rhs;
    string opSym;
};
class AndQuery:public Binary_Query{
    friend Query operator&(const Query&,const Query &);
    AndQuery(const Query& l,const Query &r):
            Binary_Query(l,r,"&"){
                cout<<"AndQuery construct"<<endl;
            }
    QueryResult eval(const TextQuery&) const;
};
QueryResult AndQuery::eval(const TextQuery& text) const{
    cout<<"AndQuery eval"<<endl;
    auto left=lhs.eval(text),right=rhs.eval(text);
    auto ret_lines=make_shared<set<line_no>>();
    set_intersection(left.begin(),left.end(),
                     right.begin(),right.end(),
                     inserter(*ret_lines,ret_lines->begin()));
    return QueryResult(rep(),ret_lines,left.get_file());
}
inline Query operator&(const Query &lhs,const Query &rhs){
    return shared_ptr<Query_base>(new AndQuery(lhs,rhs));
}
class OrQuery:public Binary_Query{
    friend Query operator|(const Query&,const Query &);
    OrQuery(const Query& l,const Query &r):
            Binary_Query(l,r,"|"){ cout<<"OrQuery construct"<<endl; }
    QueryResult eval(const TextQuery&) const;
};
inline Query operator|(const Query &lhs,const Query &rhs){
    return shared_ptr<Query_base>(new OrQuery(lhs,rhs));
}
QueryResult OrQuery::eval(const TextQuery& text) const{
    cout<<"OrQuery eval"<<endl;
    auto right=rhs.eval(text),left=lhs.eval(text);
    auto ret_lines=
        make_shared<set<line_no>>(left.begin(),left.end());
    ret_lines->insert(right.begin(),right.end());
    return QueryResult(rep(),ret_lines,left.get_file());
}

int main(){

    Query q=Query("fiery")&Query("bird")|Query("wind");
    cout<<q;

    return 0;
}