c++ 分段故障(内核转储)但无法确定

oug3syen  于 2022-11-27  发布在  其他
关注(0)|答案(1)|浏览(235)

根据我的理解,segfault通常是由无效的内存访问或泄漏引起的?但我不知道它在哪里。有没有像程序一样的方法来检查你的代码并找出它在哪里引起这个错误?请帮助。
我正在尝试用诸如insert find和predict completions这样的函数构建一个三元trie。
这是我的头文件:

#include <vector>
#include <string>
#include <list>
#include <iostream>

using namespace std;

class TNode {
        public:
                TNode* left;
                TNode* right;
                TNode* middle;
                TNode* parent;
                int freq;
                char symbol;
                bool isWord;

                TNode(const char c, bool b, const unsigned int f);
};

class DictionaryTrie {

        public:

        DictionaryTrie();

        bool insert(std::string word, unsigned int freq);

        bool find(std::string word) const;

        std::vector<std::string> predictCompletions(std::string prefix, unsigned int numcompletions);

        ~DictionaryTrie();

        private:

        TNode* root;

        void deleteAll(TNode* root) const;

        void predictHelper(TNode* n, string s, list<pair<string, unsigned int>> &  allWords) const;

};
#endif

这是我的cpp文件

#include "DictionaryTrie.hpp"
#include <list>
#include <string>
#include <vector>

using namespace std;

TNode::TNode(const char c, bool b, const unsigned int f) : symbol(c), isWord(b), freq(f), left(0), middle(0), right(0), parent(0) {}

DictionaryTrie::DictionaryTrie():root(0){}

bool DictionaryTrie::insert(string word, unsigned int freq) {
        if (word.empty())
                        return false;

        TNode* curr = root;

        if (curr == nullptr) {
                
                if (word.length() == 1)
                        root = new TNode (word[0], true, freq);
                else
                        root = new TNode(word[0], false, 0);
        }
        
        for (unsigned int i = 0; i < word.length();) {
                
                if (curr->symbol == word[i]) {

                        if (curr->middle != NULL) {
                                if (i == word.length()-1) {
                                        if(curr->isWord == false) {
                                                curr->isWord = true;
                                                curr->freq = freq;
                                                return true;
                                        }
                                        else
                                                return false;
                                }
                                else {
                                        curr = curr->middle;
                                        i++;
                                        continue;
                                }
                        }

                        else if (curr->middle == NULL) {
                                if(i!=word.length() - 1){
                                        curr->middle = new TNode(word[i+1], (i==word.length()-2),(i==word.length()-2)?freq:0);

                                        curr=curr->middle;
                                        i++;
                                        if (i==word.length()-2) return true;
                                }
                                else if (i == word.length() - 1) {
                                        if(curr->isWord == false) {
                                                curr->isWord = true;
                                                curr->freq = freq;
                                                return true;
                                        }
                                }
                                else return false;
                        }
                }
                else if(word[i] > curr->symbol) {
                        if (curr->right != NULL) {
                                curr=curr->right;
                                continue;
                        }
                        else {
                                curr->right = new TNode(word[i], (i==word.length()-1), (i==word.length()-1)?freq:0);
                               curr = curr->right;
                        if (i==word.length()-1)
                                return true;
                        }
                }
                else {
                        if (curr->left != NULL) {
                                curr = curr->left;
                                continue;
                        }
                        else {
                                curr->left = new TNode(word[i], (i==word.length()-1), (i==word.length()-1)?freq:0);
                                curr = curr->left;
                                if (i==word.length()-1)
                                        return true;
                        }
                }
        }
}
bool DictionaryTrie::find(string word) const {

        TNode* curr = root;

        if(word.length() == 0)
                return false;

        for (unsigned int i = 0; i < word.length();) {
                if (curr != NULL) {
                        if(curr->symbol == word[i]) {
                                i++;
                                if(i==word.length()) {
                                        if(curr->isWord == true)
                                                return true;
                                        else
                                                return false;
                                }
                                curr = curr->middle;
                        }
                        else if (word[i] > curr->symbol) {
                                curr = curr->right;
                        }
                        else
                                curr = curr->right;
                }
                else return false;
        }
}
vector<string> DictionaryTrie::predictCompletions(string prefix, unsigned int numCompletions) {
        vector<string> completions;

        list<pair<string, unsigned int>> allWords;

        if (numCompletions == 0 || prefix == "")
                return completions;

        TNode* curr = root;

        for (unsigned int i = 0; i < prefix.length();) {
                if (curr != NULL) {
                        if (curr->symbol == prefix[i]) {
                                i++;
                                if (i==prefix.length())
                                        break;
                                curr = curr->middle;
                        }

                        else if (prefix[i] < curr->symbol)
                                curr=curr->left;
                        else
                                curr=curr->right;
                }
                //prefix is not in trie
                else
                        return completions;
        }

        if (curr->isWord)
                allWords.push_back(make_pair(prefix, curr->freq));

        predictHelper (curr->middle, prefix, allWords);

        allWords.sort([](auto const& a, auto const& b) {

                if (a.second > b.second)
                        return true;
                if (a.second < b.second)
                        return false;
                return a.first < b.first;
                //return tie(b.second, a.first) < tie(a.second, a.first);
        });

        for(auto const& p: allWords) {
                if (completions.size() < numCompletions)
                        completions.push_back(p.first);
                else
                        break;
        }
}

void DictionaryTrie::predictHelper(TNode* n, string s, list<pair<string, unsigned int>> & allWords)const {
        if (n==NULL) return;

        else {
                string temp = s + n->symbol;

                //check if node is a valid word
                if (n->isWord == true)
                        allWords.push_back(make_pair(temp, n->freq));

        predictHelper(n->middle, temp, allWords);
        predictHelper(n->left, temp, allWords);
        predictHelper(n->right, temp, allWords);
        }
}

DictionaryTrie::~DictionaryTrie() {
        deleteAll(root);
}

void DictionaryTrie::deleteAll(TNode* n) const {

        if(!n) return;

        else {
                deleteAll(n->left);
                deleteAll(n->middle);
                deleteAll(n->right);
                delete n;
                return;
        }
}
jdzmm42g

jdzmm42g1#

当您使用调试器时,应该会看到在if (cur->symbol == word[i])处出现访问冲突。(请在家里尝试一下。)为什么会发生这种情况?
DictionaryTrie::insert中,如果curr(== root)是一个空指针,则分配一个节点并将其存储在root中,但不更新curr。然后进入for循环,并引用curr->symbol。由于curr是一个空指针,因此会出现访问冲突。
简单的解决方法是将

curr = root;

在给root赋值之后。

相关问题