使用VTK和QT多线程时的访问冲突异常

时间:2016-09-29 13:10:12

标签: c++ multithreading qt vtk qvtkwidget

我在主窗口中使用QVtkWidget。我在一个新线程中生成一个vtkRenderer,当它完成后,我发送一个signal连接slot主窗口以发送渲染器并更新QVtkWidget。

不幸的是,当我将渲染器添加到vtkRenderWindow时,我遇到了“访问冲突”异常。这是我的代码:

  

threadobject.h

class ThreadObject : public QObject
{
    Q_OBJECT

public:
    ThreadObject();
    ~ThreadObject();
signals:
    void startShowPointSignal(QString filepath);
    void endShowPointSignal(vtkRenderer* render);

public slots:
    void showPoint(QString filepath);
};
  

threadobject.cpp

#include "threadobject.h"

ThreadObject::ThreadObject(){}
ThreadObject::~ThreadObject(){}


void ThreadObject::showPoint(QString filepath)
{
    //qDebug() << filepath << QThread::currentThreadId();

    std::ifstream filestream(filepath.toLocal8Bit());
    std::string line;
    int pointCounts = 0;
    vtkSmartPointer<vtkPoints> points = vtkSmartPointer<vtkPoints>::New();
    vtkSmartPointer<vtkUnsignedCharArray> colors = vtkSmartPointer<vtkUnsignedCharArray>::New();
    colors->SetNumberOfComponents(3);
    colors->SetName("Colors");
    while (std::getline(filestream, line))
    {
        double x, y, z, r, g, b;
        std::stringstream linestream;
        linestream << line;
        linestream >> x >> y >> z >> r >> g >> b;
        points->InsertNextPoint(x, y, z);
        colors->InsertNextTuple3(r, g, b);
    }
    filestream.close();

    // Add the grid points to a polydata object
    vtkSmartPointer<vtkPolyData> polydata = vtkSmartPointer<vtkPolyData>::New();
    polydata->SetPoints(points);
    polydata->GetPointData()->SetScalars(colors);

    vtkSmartPointer<vtkVertexGlyphFilter> glyphFilter = vtkSmartPointer<vtkVertexGlyphFilter>::New();
    glyphFilter->SetInputData(polydata);
    glyphFilter->Update();

    vtkSmartPointer<vtkCleanPolyData> cleaner = vtkSmartPointer<vtkCleanPolyData>::New();
    cleaner->SetInputConnection(glyphFilter->GetOutputPort());
    cleaner->Update();

    // Create a mapper and actor
    vtkSmartPointer<vtkPolyDataMapper> pointMapper = vtkSmartPointer<vtkPolyDataMapper>::New();
    pointMapper->SetInputConnection(cleaner->GetOutputPort());
    vtkSmartPointer<vtkActor> pointActor = vtkSmartPointer<vtkActor>::New();
    pointActor->SetMapper(pointMapper);

    // Create a renderer
    vtkSmartPointer<vtkRenderer> renderer = vtkSmartPointer<vtkRenderer>::New();

    // Add the actor to the scene
    renderer->AddActor(pointActor);
    renderer->ResetCamera();
    renderer->SetBackground(0.1, 0.1, 0.1);

    //send the renderer back to my qt window
    emit endShowPointSignal(renderer);      
    //delete this;
}
  

mainwindow.cpp

void MainWindow::test()
{
    QThread* thr1 = new QThread();
    ThreadObject* obj = new ThreadObject();
    obj->moveToThread(thr1);
    connect(obj, SIGNAL(startShowPointSignal(QString)), obj, SLOT(showPoint(QString)), Qt::QueuedConnection);
    connect(obj, SIGNAL(endShowPointSignal(vtkRenderer*)), this, SLOT(updateVTK(vtkRenderer*)), Qt::QueuedConnection);
    thr1->start();
    obj->startShowPointSignal("D:/WorkSpace/VS Project/Plane/Datas/1.xyz");
}

void MainWindow::updateVTK(vtkRenderer* renderer)
{
    double test[3];
    renderer->GetBackground(test);

    vtkSmartPointer<vtkRenderWindow> renderWindow = vtkSmartPointer<vtkRenderWindow>::New();
    renderWindow->AddRenderer(renderer); // I GOT THE ACCESS VIOLATION THIS LINE!!!!!

    vtkSmartPointer<vtkRenderWindowInteractor> renderWindowInteractor = vtkSmartPointer<vtkRenderWindowInteractor>::New();
    vtkSmartPointer<vtkInteractorStyleTrackballCamera> istyle = vtkSmartPointer<vtkInteractorStyleTrackballCamera>::New();
    renderWindowInteractor->SetInteractorStyle(istyle);
    renderWindowInteractor->SetRenderWindow(renderWindow);

    ui.qvtkWidget->SetRenderWindow(renderWindow);

    renderWindow->Render();
    renderWindowInteractor->Start();
}

我试图将所有代码放在mainwindow中,而不是使用新的线程,它很有趣。

使用QT线程时有什么问题吗?或者其他我没注意到的东西?

我是QT的新手,QT和VTK的版本是5.7和7.

1 个答案:

答案 0 :(得分:1)

好的,最后我发现这是一个非常愚蠢的问题。我使用vtkSmartPointer创建了渲染器,但信号和插槽的参数是常规指针。将参数更改为vtkSmartPointer格式后,问题就消失了。