Вызов pthread_create внутри функции-члена? - PullRequest
0 голосов
/ 15 июня 2011

Я создал файл widget.h, содержащий сокращения pthread_function, и я хотел вызвать его в функции-члене destroyWidget этого класса Widget в widget.cpp.но всегда показывает ошибку.Я покажу .cpp и .h файл.

файл widget.h


class Widget
{
public:
Widget();
void createWidget(int x,int y,int w,int h);
void showWidget();
int wid;
pthread_t thread;
int *incomingval,id;
void join();
Window win;
XEvent evt;
private:
void* destroyWidget(void* ptr);
Display *disp;
int screenNumber;
unsigned long white;
unsigned long black;
long eventMask;
GC gc;
int tbit;
int *incoming,val;
};

теперь widget.cpp


Widget::Widget()
{
disp=XOpenDisplay( NULL );
screenNumber=DefaultScreen(disp);
white=WhitePixel(disp,screenNumber);
black=BlackPixel(disp,screenNumber);
eventMask=StructureNotifyMask;
tbit=0;
}

void Widget::createWidget(int x,int y,int w,int h)
{
wid=w;
win= XCreateSimpleWindow(disp,DefaultRootWindow(disp),x,y,w,h,1,white,black);
}

void Widget::showWidget()
{
XMapWindow(disp,win);
XFlush(disp);
gc=XCreateGC(disp,win,0,NULL);
XSetForeground(disp,gc,white);
XDrawLine(disp,win,gc,wid-10,0,wid,10);
XDrawLine(disp,win,gc,wid-10,10,wid,0);
//calling the thread function
pthread_create( &thread, NULL, destroyWidget, this);
}

void Widget::join()
{
pthread_join( thread, NULL);
}
void* Widget::destroyWidget(void* ptr)
{
Widget* mw = static_cast(ptr);
eventMask=ButtonPressMask|ButtonReleaseMask;
XSelectInput(disp,win,eventMask);
do{
printf("id= %d",id);
XNextEvent(disp,&evt);
}while(evt.type!=ButtonRelease);
XDestroyWindow(disp,win);
XCloseDisplay(disp);
return NULL;
}

теперь main.cppфайл



#include "widget.h"
#include
int main()
{
Widget* w=new Widget();
Widget* n=new Widget();
n->createWidget(20,20,150,150);
w->createWidget(50,50,250,250);
n->showWidget();
w->showWidget();
n->join();
w->join();
return 0;
}

ошибка

widget.cpp: In member function ‘void Widget::showWidget()’:
widget.cpp:44:51: error: argument of type ‘void* (Widget::)(void*)’ does not match ‘void* (*)(void*)’

Ответы [ 3 ]

7 голосов
/ 15 июня 2011

Проблема в том, что pthread_create - это функция в стиле C;Вы должны дать ему указатель на функцию.Widget::destroyWidget() является указателем на член -функцию.(Помните, что нестатические функции-члены всегда имеют неявный аргумент this, который pthread_create не знает, как его предоставить.)

См. Ответы на этот вопрос для некоторых возможных решений: Функция pthread из класса .

0 голосов
/ 15 июня 2011

Третий аргумент pthread_create имеет подпись (в C ++):

extern "C" void* (*pointerToFunction)( void* );

Вы пытаетесь передать ему адрес функции-члена:

void* (Widget::*pointerToMemberFunction)( void* );

Сигнатуры несовместимы: вторая требует объекта, для которого она вызывается, и не является extern "C".

. Самый простой способ справиться с этим - использовать boost::thread со всей поддержкой функциональных объектов.В противном случае вы можете определить что-то вроде следующего:

struct AbstractTask
{
    virtual ~AbstractTask() {}
    virtual void* run() = 0;
};

template<typename T, void* (T::*ptr)()>
class Task
{
    T* myObject;
public:
    Task( T* object ) : myObject( object ) {}
    virtual void* run()
    {
        return (myObject->*ptr)();
    }
};

extern "C" void* taskRunner( void* arg )
{
    std::auto_ptr<AbstractTask> p( static_cast<AbstractTask*>( arg ) );
    return p->run();
}

pthread_t taskStarter( AbstractTask* obj )
{
    pthread_t result;
    pthread_create( &result, NULL, &taskRunner, obj );
    return result;
}

Чтобы запустить поток, вы затем вызываете:

thread = taskStarter( new Task<Widget, &Widget::destroyWidget>( this ) );

(Это из памяти, из более раннего проекта, так чтов этом могут быть некоторые опечатки, но вы поняли, и, возможно, вы захотите добавить обработку ошибок в taskStarter.)

0 голосов
/ 15 июня 2011

Как сказал Оли, вы не можете использовать функцию-член, когда функция в стиле C ожидает «нормального» указателя на функцию.Однако вы можете создать отдельную функцию, которая вызывает ваш метод destroyWidget ().

Примерно так:

void* start_routine(void* arg)
{
    Widget* widget = static_cast<Widget* >(arg);

    widget->destroyWidget();

    return NULL;
}

void Widget::showWidget()
{
    pthread_create(&thread, NULL, &start_routine, this);
}

void Widget::destroyWidget()
{
    // your code
}
...