c++ - Qt QString克隆Segmentation Fault

标签 c++ qt pointers qstring

我正在使用 Qt Creator 构建我的第一个 Qt 应用程序,一切都很顺利,直到我开始从一条看似无害的线路中收到一个奇怪的 SIGSEGV。

这是错误:

Program received signal SIGSEGV, Segmentation fault. 0x0804e2fe in QBasicAtomicInt::ref (this=0x0) at /usr/lib/qt/include/QtCore/qatomic_i386.h:120

通过回溯 gdb 上的异常,我发现当我返回我的属性时,一个简单的 getter 将 NULL 指针传递给克隆构造函数。

回溯输出:

(gdb) backtrace
#0 0x0804e2fe in QBasicAtomicInt::ref (this=0x0) at /usr/lib/qt/include/QtCore/qatomic_i386.h:120
#1 0x0804eb1b in QString (this=0xbfcc8e48, other=@0xbfcc8e80) at /usr/lib/qt/include/QtCore/qstring.h:712
#2 0x0805715e in Disciplina::getId (this=0xbfcc8e7c) at disciplina.cpp:13
[...]

检查传递给 QString 构造函数的指针:

(gdb) x 0xbfcc8e80
0xbfcc8e80: 0x00000000

这是 disciplina.cpp:13

QString Disciplina::getId()
{
    return id;
}

因此,所有指向接收空指针的 QString 的复制构造函数,这对我来说毫无意义。 id 被声明为私有(private) QString。

private:
    QString id;

好吧,我不知道会发生什么,我的调试技能也仅限于此,所以如果有人能提出一个想法,我会非常高兴。

谢谢。

编辑

根据要求提供更多代码。

disciplina.h

#ifndef DISCIPLINA_H
#define DISCIPLINA_H
#include <QString>
#include <QMap>
#include "curso.h"
#include "turma.h"

class Curso;

class Turma;

class Disciplina
{
private:
    unsigned short int serie;
    QString id;
    QString nome;
    Curso* curso;
    QMap<unsigned int, Turma*> turmas;    
public:
    Disciplina(QString id, Curso* curso, QString nome, unsigned short int serie);

    QString getId();
    const Curso getCurso();
    QString getNome();
    void setNome(QString nome);
    void addTurma(Turma* t, unsigned int id);
    QMap<unsigned int, Turma*> getTurmas();
};

#endif // DISCIPLINA_H

disciplina.cpp

#include "disciplina.h"

Disciplina::Disciplina(QString id, Curso* curso, QString nome, unsigned short int serie)
{
    this->id = id;
    this->curso = curso;
    this->nome = nome;
    this->serie = serie;
}

QString Disciplina::getId()
{
    return id;
}

const Curso Disciplina::getCurso()
{
    const Curso c(*this->curso);
    return c;
}

QString Disciplina::getNome()
{
    return this->nome;
}

void Disciplina::setNome(QString nome)
{
    this->nome = nome;
}

void Disciplina::addTurma(Turma* t, unsigned int id)
{
    this->turmas.insert(id, t);
}

QMap<unsigned int, Turma*> Disciplina::getTurmas()
{
    return this->turmas;
}

调用函数(我将其分解以便于调试)

Disciplina*
MainWindow::getSelectedDisciplina()
{
    if(ui->disciplinaTurma->count() > 0 && currentCurso)
    {
        QMap<QString, Disciplina*> qm(currentCurso->getDisciplinas());
        QString key = ui->disciplinaTurma->itemText(ui->disciplinaTurma->currentIndex());
        Disciplina* d = qm[key];
        QMessageBox::information(this, d->getId(), d->getNome());
        return d;
    }
    else
        return NULL;
}

已解决

插入 map 的 Disciplina 对象超出范围,因此被删除。因为,正如 Jacinto 指出的那样,当您尝试访问一个不存在的键时,Map 创建了一个原始值,看起来对象就在那里。

感谢 Jacintosth 的帮助。

最佳答案

在 c++ 的映射中,如果当您尝试通过其键访问它时该元素不存在,它只会为您创建一个。您正试图在这里做同样的事情,如果 QMap 以同样的方式工作,这就是导致您的段错误的原因。

您应该做的是在访问之前测试 key 在 map 中的存在。

编辑:对于 C++ 纯粹主义者,如果我说得对,请告诉我。我知道在实践中在访问它之前进行测试更安全,但我不知道“它为你创建一个”这样的措辞是否是一个很好的表达方式。它可能只是返回内存中这样一个值所在的空间;我不知道它是否真的会调用默认构造函数。

关于c++ - Qt QString克隆Segmentation Fault,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1861964/

相关文章:

c++ - C++中指针结构的运算符重载

c++ - 是否可以调用一个以 const 右值作为参数的函数?

html - QTextBrowser 不显示来自 html 文件的图像(Windows 7)

c++ - 重新排序 gnu autotools 链接器标志

c++ - QProperties 中的共享指针列表

c++ - Qt QLCDNumber time_t 定时器

c - 数组的指针算术

c - 这个指向指针的指针是多余的吗?

c++ - main : linking fortran with C++的多重定义

c++ - 如何编译使用 std::chrono 和日期的 CUDA 代码?