从库(DLL)函数访问ui指针

Access ui pointer from a library (DLL) function

本文关键字:访问 ui 指针 函数 DLL 从库      更新时间:2023-10-16

嗨,我想从一个非成员函数更新ui。除了将"this"指针作为我的非成员传递之外,任何帮助都是来自库的回调。

以下是我的代码:

主窗口.cpp

static void callback(QString result) 
{
ui->textBrowser->append(result);
}
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
ui->setupUi(this);
}
MainWindow::~MainWindow()
{
delete ui;
}
void MainWindow::on_pushButton_clicked()
{
int a=1,b=2;
QLibrary myLib("myaddlib");
myLib.load();
add = (myadd)myLib.resolve("add_function");
add(callback, a, b);
}

因此,我所需要的就是能够从非成员回调中将数据附加到UI中。请帮帮我。

编辑:我不允许修改我的回调

基于注释的EDIT:callback将在另一个线程中调用,因此从那里调用小部件方法会产生错误'cannot send events to objects owned by a different thread'

根据您的评论

"我尝试创建全局指针Ui::MainWindow*myui并分配构造函数中的myui到ui,但它会引发以下错误:"无法将事件发送到其他线程拥有的对象"。"

问题是,您的callback在错误的线程中被调用。这里有一个解决方案:

  1. 创建指向实际主窗口实例的全局指针,我们称之为MainWindow *mainWinInstance;。还要注意的是,主窗口实例的使用寿命将比回调所针对的库/线程长,或者如果即使在主窗口被破坏后callback也有被调用的风险,则使用QPointer

  2. 将以下插槽方法添加到MainWindow:

    MainWindow::appendText(const QString &text) { // use const ref for efficiency
        ui->textBrowser->append(result);
    }
    
  3. 更改callback以使用排队连接类型的QMetaObject::invokeMethod调用该方法:

    static void callback(QString result) {
        bool r = QMetaObject::invokeMethod(mainWinInstance, 
                                           "appendText", 
                                           Qt::QueuedConnection,
                                           Q_ARG(QString, result));
        Q_ASSERT(r); // should only fail if there's a mistake in above code
    }
    

    使用CCD_ 10是非常重要的。它将把方法调用放在目标对象的正确线程的事件队列中,并立即返回。然后,目标线程的事件循环将执行实际调用。

您需要在一个单独的头中创建一个接口:

itextbrowseraccessor.h
class ITextBrowserAccessor
{
public:
  void appendText(const QString& text) = 0;
}

ITextBrowserAccessor继承MainWindow并实现方法:

void MainWindow::appendText(const QString& text)
{
   ui->textBrowser->append(text);
}

在库源文件中包含itextbrowseraccessor.h。将指向接口的指针传递给回调。

static void callback(ITextBrowserAccessor* accessor, QString result) 
{
  accessor->appendText(result);
}

您可以尝试使用QApplication::topLevelWidgets()查找主窗口。

找到主窗口后,使用mainWindow->findChild<QTextBrowser*>()查找浏览器小部件,然后根据需要修改其内容。

编辑:
示例:

Q_ASSERT(QApplication::topLevelWidgets() == 1);
QWidget* mainWindow = QApplication::topLevelWidgets().first();
QTextEditor* editor = mainWindow->findChild<QTextBrowser*>();
Q_ASSERT(editor != NULL);
editor->append(QLatin1String("abc"));

这里提到的所有事情对我都不起作用。通过创建一个具有公共函数的伪Qobject来向主窗口发送信号,从而解决了这个问题。创建这个伪qobject的一个实例,并在回调中调用公共函数。

附言:我有一个限制,不能修改我的回调,因此上面的解决方案。