理論上 QWidget及其派生類是不可以用 Q_DECLARE_METATYPE(QWidget); 的,因?yàn)槟?寫(xiě)下這樣的代碼編譯通不過(guò)
會(huì)報(bào)告一個(gè) :error C2248: 'QWidget::QWidget' : cannot access private member declared in class 'QWidget'
原因在于 QWidget里 有 Q_DISABLE_COPY(QWidget),所以把拷貝構(gòu)造和等號(hào)賦值都私有化了,但是Q_DECLARE_METATYPE里面需要訪問(wèn)拷貝構(gòu)造
因?yàn)?Q_DECLARE_METATYPE 需要構(gòu)建一個(gè) Constructor
代碼如下:
template<typename T>
void *qMetaTypeConstructHelper(const T *t)
{
if (!t)
return new T;
return new T(*static_cast(t)); //此處需要訪問(wèn)拷貝構(gòu)造函數(shù),從而造成編譯錯(cuò)誤
}
即使實(shí)際運(yùn)行時(shí)候我并不需要拷貝構(gòu)造,但是編譯時(shí)是需要,所以無(wú)法編譯。
有個(gè)方法可以欺騙編譯器,就是給QWidget類寫(xiě)一個(gè) qMetaTypeConstructHelper的特化版本
聲明成 Static 是為了避免link error
代碼如下:
template <>
static void *qMetaTypeConstructHelper(const QWidget *t)
{
return new QWidget();
}
Q_DECLARE_METATYPE(QWidget);
于是這樣就可以創(chuàng)建 QWidget了,代碼如下:
QWidget *w= static_cast< QWidget*>(QMetaType::construct(qMetaTypeId()));
//或者
qRegisterMetaType("QWidget");
QWidget *w= static_cast< QWidget*>(QMetaType::construct(QMetaType::type("QWidget")));
這種方法是可以欺騙編譯器,并且可以達(dá)到動(dòng)態(tài)創(chuàng)建的目的,但是并不等于我推薦這樣的做法,實(shí)際上這樣做并不比你為每個(gè)類名稱寫(xiě)if else的動(dòng)態(tài)創(chuàng)建簡(jiǎn)單多少。。。