2

这个问题的一些背景是我之前的问题: 非成员函数指针作为 API 中对成员函数的回调(它可能无关紧要)。

回调启动一个写入一些数据的线程。还有另一个线程读取相同的数据,这会导致一些崩溃。我刚刚参加了多线程速成课程(感谢 SO),这是我试图保证作者和读者不会同时访问数据的尝试。我正在使用来自 Qt (QReadWriteLock) 的一些互斥机制。

#include <QSharedPointer>
#include <QReadWriteLock>

Class MyClass
{
public:
  MyClass();
  bool open();
  float getData();
  void streamCB(void* userdata);

protected:
  float private_data_;
  QSharedPointer<QReadWriteLock> lock_;
};

// callback wrapper from non-member C API to member function void
__stdcall streamCBWrapper(void* userdata)
{
   static_cast<MyClass*>(userdata)->streamCB(userdata);
}

// constructor
MyClass::MyClass()
{
  lock_ = QSharedPointer<QReadWriteLock>(new QReadWriteLock());
  lock_->lockForWrite();
  private_data_ = getData();
  lock_->unlock();
}

// member callback
void MyClass:streamCB(void* userdata)
{
  float a = getData(); 
  lock_->lockForWrite(); //*** fails here
  private_data_ = a;
  lock_->unlock();
}

运行程序时出现分段错误。VS 调试器Access violation writing location 0x00fedbed.在我标记的行上说//*** fails here. 锁在构造函数中起作用,但在回调中不起作用。

知道出了什么问题吗?我应该看什么?(以及如何改进我的问题)谢谢!

其他相关线程 无法访问在类'QReadWriteLock'Error 1 error C2248中声明的私有成员:'QReadWriteLock::QReadWriteLock'(我使用了QSharedPointer建议)

编辑1: 回调设置

bool MyClass::open()
{
  // stuffs
  int mid = 0;
  set_stream_callback(&streamCBWrapper, &mid);
  // more stuffs
  return true;
}

编辑2: 感谢您的所有建议。所以我的错误可能根本不是由于互斥锁,而是因为我对 API 缺乏了解?我很困惑.. 这是 set_stream_callback 的 API 文档。

typedef void (__stdcall *STREAM_CALLBACK)(void *userdata);

/*! @brief Register a callback to be invoked when all subscribed fields have been updated
 *
 *  @param streamCB pointer to callback function
 *  @param userdata pointer to private data to be passed back as argument to callback
 *  @return 0 if successful, error code otherwise
 */
__declspec(dllimport) int __stdcall set_stream_callback(
    STREAM_CALLBACK streamCB, void *userdata);
4

2 回答 2

1

为什么需要足够的代码示例的好例子。

如果我正确解释了您的回调语法,

set_stream_callback(&streamCBWrapper, &mid);

设置streamCBWrapper为回调函数,while&mid是指向userdata.

在回调中,您实际上是在将指针转换intMyClass,然后尝试访问不存在对象的成员变量。

确保将一个实例传递MyClass给您的回调。我想这就是this你的情况。

于 2012-11-13T16:40:20.137 回答
1

对我来说,这听起来基本上像是一个线程问题。由于无论如何您都在使用 Qt 互斥,您可能会考虑使用 Qt 的线程机制并在线程之间发送信号和插槽。只要您遵循此处此处的建议,它们的文档就很好并且易于使用。

于 2012-11-13T18:16:57.617 回答