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