Ошибка сегментации, вызванная пустой ссылкой в заголовочном файле c ++

0

Может ли кто-нибудь узнать, что не так со следующей частью кода:

private:
     TUcdFileReader& operator = (const TUcdFileReader& r) { Fail; return *((TUcdFileReader *) 0); }
     TUcdFileReader(const TUcdFileReader& r) { Fail; }

Это относится к файлу заголовка в библиотеке ac (SNAP), который я использую в своем коде. Когда я компилирую свой код в системе на основе Linux, появляется следующее предупреждение:

Snap-2.1/glib-core/unicode.h(1678): warning #327: NULL reference is not allowed
                TUcdFileReader& operator = (const TUcdFileReader& r) { Fail; return *((TUcdFileReader *) 0); }

Что в целом привело меня к ошибке сегментации следующим образом: (и я подозреваю, что это из-за вышеупомянутого предупреждения)

WARNING: Job died due to SIGSEGV - Invalid memory reference

Я не пишу весь файл заголовка, так как он длинный и вызывает путаницу, но это его адрес: http://snap.stanford.edu/snap/doc/snapuser-ref/dd/d90/unicode_8h_source.html Здесь я написал большую часть файла заголовка, который использует TUcdFileReader:

protected:

class TUcdFileReader
{
protected:
    TChA buf;
public:
    TChA comment; // contains '#' and everything after it
protected:
    FILE *f;
    int putBackCh;
    int GetCh() {
        if (putBackCh >= 0) { int c = putBackCh; putBackCh = EOF; return c; }
        return fgetc(f); }
    void PutBack(int c) { Assert(putBackCh == EOF); putBackCh = c; }
    // Returns 'false' iff the EOF was encountered before anything was read.
    bool ReadNextLine() {
        buf.Clr(); comment.Clr();
        bool inComment = false, first = true;
        while (true) {
            int c = GetCh();
            if (c == EOF) return ! first;
            else if (c == 13) {
                c = GetCh(); if (c != 10) PutBack(c);
                return true; }
            else if (c == 10) return true;
            else if (c == '#') inComment = true;
            if (! inComment) buf += char(c);
            else comment += char(c); }
            /*first = false;*/}
private:
    TUcdFileReader& operator = (const TUcdFileReader& r) { Fail; return *((TUcdFileReader *) 0); }
    TUcdFileReader(const TUcdFileReader& r) { Fail; }
public:
    TUcdFileReader() : f(0) { }
    TUcdFileReader(const TStr& fileName) : f(0), putBackCh(EOF) { Open(fileName); }
    void Open(const TStr& fileName) { Close(); f = fopen(fileName.CStr(), "rt"); IAssertR(f, fileName); putBackCh = EOF; }
    void Close() { putBackCh = EOF; if (f) { fclose(f); f = 0; }}
    ~TUcdFileReader() { Close(); }
    bool GetNextLine(TStrV& dest) {
        dest.Clr();
        while (true) {
            if (! ReadNextLine()) return false;
            TStr line = buf; line.ToTrunc();
            if (line.Len() <= 0) continue;
            line.SplitOnAllCh(';', dest, false);
            for (int i = 0; i < dest.Len(); i++) dest[i].ToTrunc();
            return true; }}
    static int ParseCodePoint(const TStr& s) {
        int c; bool ok = s.IsHexInt(true, 0, 0x10ffff, c); IAssertR(ok, s); return c; }
    static void ParseCodePointList(const TStr& s, TIntV& dest, bool ClrDestP = true) { // space-separated list
        if (ClrDestP) dest.Clr();
        TStrV parts; s.SplitOnWs(parts);
        for (int i = 0; i < parts.Len(); i++) {
            int c; bool ok = parts[i].IsHexInt(true, 0, 0x10ffff, c); IAssertR(ok, s);
            dest.Add(c); } }
    static void ParseCodePointRange(const TStr& s, int& from, int &to) { // xxxx or xxxx..yyyy
        int i = s.SearchStr(".."); if (i < 0) { from = ParseCodePoint(s); to = from; return; }
        from = ParseCodePoint(s.GetSubStr(0, i - 1));
        to = ParseCodePoint(s.GetSubStr(i + 2, s.Len() - 1)); }
};

Кроме того, это единственное место, где вызывается TUcdFileReader:

class TSubcatHelper
    {
    public:
        bool hasCat; TUniChSubCategory subCat;
        TStrH invalidCatCodes;
        TUniChDb &owner;

        TSubcatHelper(TUniChDb &owner_) : owner(owner_) { }

        void ProcessComment(TUniChDb::TUcdFileReader &reader)
        {
....
Теги:
nullreferenceexception
segmentation-fault

1 ответ

0
Лучший ответ

Класс TUcdFileReader определен конкретно как НЕ имеющий назначение или конструкторы копирования, т.е.

private:
    TUcdFileReader& operator = (const TUcdFileReader& r) 
               { Fail; return *((TUcdFileReader *) 0); }
    TUcdFileReader(const TUcdFileReader& r) { Fail; }

похоже, что они предназначены для отказа.

Где-то в коде они вызываются - если вы замените их:

private:
    TUcdFileReader& operator = (const TUcdFileReader& r); //**not implemented** 
    TUcdFileReader(const TUcdFileReader& r);              //**not implemented**  

то вы получите ошибку компоновщика. Лучшее исправление будет (если возможно) для их реализации.

  • 0
    Единственное место, где вызывается TUcdFileReader: void ProcessComment (TUniChDb :: TUcdFileReader & reader). Я обновляю эту часть в своем вопросе. Спасибо за подсказку, но я не понял, как это исправить.
  • 1
    Предполагая, что C ++ 11, вы даже можете пометить свои нежелательные конструкторы = delete следующим образом: TUcdFileReader& operator = (const TUcdFileReader& r) = delete; чтобы получить ошибку компиляции, а не ошибку компоновщика (которую вы заметили ранее).
Показать ещё 1 комментарий

Ещё вопросы

Сообщество Overcoder
Наверх
Меню