c++ - 在预期非成员函数指针时传递成员函数指针?

标签 c++ class c++11 callback portaudio

我目前在将我的成员函数回调函数传递给库 portaudio 定义的非成员函数时遇到一些问题。

我的类是这样定义的:

class record {
public:
    record();
    void start_record();
    int recordCallback(const void *inputBuffer, void *outputBuffer,
                       unsigned long framesPerBuffer,
                       const PaStreamCallbackTimeInfo* timeInfo,
                       PaStreamCallbackFlags statusFlags, void *userData);
private:
    PaStreamParameters  inputParameters,
                        outputParameters;
    PaStream*           stream;
    PaError             err = paNoError;
    paTestData          data;
    int                 totalFrames;
    int                 numSamples;
    int                 numBytes;
    SAMPLE              max, val;
    double              average;
};

start_record() 中,我将成员函数传递给非成员函数。这意味着我将类记录(回调)的成员函数传递给非成员函数 Pa_OpenStream ().

err = Pa_OpenStream(
          &this->stream,
          &this->inputParameters,
          NULL,                  /* &outputParameters, */
          SAMPLE_RATE,
          FRAMES_PER_BUFFER,
          paClipOff,      /* we won't output out of range samples so don't bother clipping them */
          this->recordCallback,
          &data );
if( err != paNoError )
{
    std::cout << "Something wrong  - open_stream check" << std::endl;
    exit(1);
}

portaudio 需要类型的函数指针

int (*)(const void*, void*, long unsigned int, const PaStreamCallbackTimeInfo*, PaStreamCallbackFlags, void*)
and not

而不是

int (record::)(const void*, void*, long unsigned int, const PaStreamCallbackTimeInfo*, PaStreamCallbackFlags, void*) 

一个快速的解决方案是在类作用域之外定义函数,并使它使用和在类中定义的变量成为全局变量,但我想避免这种情况。那么我该如何处理呢?

最佳答案

duplicate你的previous question列出了一些想法,但缺少具体代码,所以这里有一个示例。

首先,通过添加静态回调函数来修改类记录:

static int myCallback(const void *inputBuffer, void outputBuffer, unsigned long framesPerBuffer, const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags, void *userData);

请注意,这与您现有的回调具有相同的签名,只是它是一个静态函数,因此您可以将它传递给 Pa_OpenStream

定义该函数以调用您的成员函数回调:

int record::myCallback(const void *inputBuffer, void outputBuffer, unsigned long framesPerBuffer, const PaStreamCallbackTimeInfo* timeInfo, PaStreamCallbackFlags statusFlags, void *userData)
{
    record *pThis = (record *)userData;
    return pThis->recordCallback(inputBuffer, outputBuffer, framesPerBuffer, timeInfo, statusFlags, nullptr);
}

我们在这里传递 nullptr 作为用户数据指针,因为我们需要的所有用户数据都保存在类中。由于此回调仅在内部使用,您只需从此处和函数中删除最后一个参数即可。

当您注册回调时,传入新的回调,并将this 作为用户数据参数。

err = Pa_OpenStream(
      &this->stream,
      &this->inputParameters,
      NULL,                  /* &outputParameters, */
      SAMPLE_RATE,
      FRAMES_PER_BUFFER,
      paClipOff,      /* we won't output out of range samples so don't bother clipping them */
      &this->myCallback,
      this);

然后在 recordCallback 中,您可以像往常一样访问您类(class)的所有成员。

关于c++ - 在预期非成员函数指针时传递成员函数指针?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47065089/

相关文章:

c++ - cl::Program::Sources 在哪里定义的?

c++ - GCC模板问题

python - 从类实例python中删除属性

c++ - 用于从头文件创建类主体的智能工具(插件?)

javascript - 访问任意父类(super class)中声明的属性

c++ - constexpr 静态数据成员给出 undefined reference 错误

c++ - 通过组合实现仿函数重载

c++ - 错误 : expected constructor, 析构函数,或 '(' token 之前的类型转换?

c++ - 在哪里实现哈希函数?

C++ 模板重载 - 调用了错误的函数