Advertisement
Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- #pragma once
- #include <iostream>
- #include <vector>
- #include <set>
- #include <map>
- #include <algorithm>
- #include "string_processing.h"
- #include "document.h"
- using namespace std::string_literals;
- const int MAX_RESULT_DOCUMENT_COUNT = 5;
- const double ACCURACY = 1e-6;
- class SearchServer {
- public:
- template <typename StringContainer>
- explicit SearchServer(const StringContainer& stop_words);
- explicit SearchServer(const std::string& stop_words_text);
- void AddDocument(int document_id, const std::string& document, DocumentStatus status,
- const std::vector<int>& ratings);
- template <typename DocumentPredicate>
- std::vector<Document> FindTopDocuments(const std::string& raw_query,
- DocumentPredicate document_predicate) const;
- std::vector<Document> FindTopDocuments(const std::string& raw_query, DocumentStatus status) const;
- std::vector<Document> FindTopDocuments(const std::string& raw_query) const;
- int GetDocumentCount() const;
- std::vector<int>::const_iterator begin() const;
- std::vector<int>::const_iterator end() const;
- std::tuple<std::vector<std::string>, DocumentStatus> MatchDocument(const std::string& raw_query,
- int document_id) const;
- const std::map<std::string, double>& GetWordFrequencies(int document_id) const;
- void RemoveDocument(int document_id);
- std::vector<int> duplicates;
- private:
- struct DocumentData {
- int rating;
- DocumentStatus status;
- };
- const std::set<std::string> stop_words_;
- std::map<int, std::map<std::string, double>> word_freqs_in_document;
- std::map<int, DocumentData> documents_;
- std::vector<int> document_ids_;
- bool IsStopWord(const std::string& word) const;
- static bool IsValidWord(const std::string& word);
- std::vector<std::string> SplitIntoWordsNoStop(const std::string& text) const;
- static int ComputeAverageRating(const std::vector<int>& ratings);
- struct QueryWord {
- std::string data;
- bool is_minus;
- bool is_stop;
- };
- QueryWord ParseQueryWord(const std::string& text) const;
- struct Query {
- std::set<std::string> plus_words;
- std::set<std::string> minus_words;
- };
- Query ParseQuery(const std::string& text) const;
- double ComputeWordInverseDocumentFreq(const std::string& word) const;
- template <typename DocumentPredicate>
- std::vector<Document> FindAllDocuments(const Query& query,
- DocumentPredicate document_predicate) const;
- };
- template <typename StringContainer>
- SearchServer::SearchServer(const StringContainer& stop_words)
- : stop_words_(MakeUniqueNonEmptyStrings(stop_words))
- {
- if (!std::all_of(stop_words_.begin(), stop_words_.end(), IsValidWord)) {
- throw std::invalid_argument("Some of stop words are invalid"s);
- }
- }
- template <typename DocumentPredicate>
- std::vector<Document> SearchServer::FindTopDocuments(const std::string& raw_query,
- DocumentPredicate document_predicate) const {
- const auto query = ParseQuery(raw_query);
- auto matched_documents = FindAllDocuments(query, document_predicate);
- std::sort(matched_documents.begin(), matched_documents.end(),
- [](const Document& lhs, const Document& rhs) {
- if (std::abs(lhs.relevance - rhs.relevance) < ACCURACY) {
- return lhs.rating > rhs.rating;
- }
- else {
- return lhs.relevance > rhs.relevance;
- }
- });
- if (matched_documents.size() > MAX_RESULT_DOCUMENT_COUNT) {
- matched_documents.resize(MAX_RESULT_DOCUMENT_COUNT);
- }
- return matched_documents;
- }
- template <typename DocumentPredicate>
- std::vector<Document> SearchServer::FindAllDocuments(const Query& query,
- DocumentPredicate document_predicate) const {
- std::map<int, double> document_to_relevance;
- for (const std::string& word : query.plus_words) {
- bool flag = 0;
- for (const auto [id, doc] : word_freqs_in_document) {
- if (doc.count(word) > 0) {
- flag = 1;
- break;
- }
- }
- if (flag == 0) {
- continue;
- }
- const double inverse_document_freq = ComputeWordInverseDocumentFreq(word);
- for (const auto& [document_id, words] : word_freqs_in_document) {
- if (words.count(word) == 0)
- continue;
- const auto& document_data = documents_.at(document_id);
- if (document_predicate(document_id, document_data.status, document_data.rating)) {
- document_to_relevance[document_id] += words.at(word) * inverse_document_freq;
- }
- }
- }
- for (const std::string& word : query.minus_words) {
- bool flag = 0;
- for (const auto [id, doc] : word_freqs_in_document) {
- if (doc.count(word) > 0) {
- flag = 1;
- break;
- }
- }
- if (flag == 0) {
- continue;
- }
- for (const auto& [document_id, words] : word_freqs_in_document) {
- if (words.count(word)>0)
- document_to_relevance.erase(document_id);
- }
- }
- std::vector<Document> matched_documents;
- for (const auto [document_id, relevance] : document_to_relevance) {
- matched_documents.push_back(
- { document_id, relevance, documents_.at(document_id).rating });
- }
- return matched_documents;
- }
- void MatchDocuments(const SearchServer& search_server, const std::string& raw_query);
- void FindTopDocuments(const SearchServer& search_server, const std::string& raw_query);
Advertisement
Add Comment
Please, Sign In to add comment
Advertisement