青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Cpper
C/C++高級工程師 Android高級軟件工程師 IT集成工程師 音頻工程師 熟悉c,c++,java,c#,py,js,asp等多種語言 程序猿
QVariant是一種可以存儲不同類型的數據結構,在很多場合這是很有用得
為了達到這種目的,可以想象,該對象應該存儲對象的類型信息,數據信息以及其他輔助詳細
考慮用途,這種對象必須支持對不同對象的存儲,對存儲類型的檢測以及取對象三個功能
1.對象的存儲
代碼見下:
    QVariant(Type type);
    QVariant(
int typeOrUserType, const void *copy);
    QVariant(
int typeOrUserType, const void *copy, uint flags);
    QVariant(
const QVariant &other);

#ifndef QT_NO_DATASTREAM
    QVariant(QDataStream 
&s);
#endif

    QVariant(
int i);
    QVariant(
uint ui);
    QVariant(qlonglong ll);
    QVariant(qulonglong ull);
    QVariant(
bool b);
    QVariant(
double d);
    QVariant(
float f) { d.is_null = false; d.type = QMetaType::Float; d.data.f = f; }
#ifndef QT_NO_CAST_FROM_ASCII
    QT_ASCII_CAST_WARN_CONSTRUCTOR QVariant(
const char *str);
#endif

    QVariant(
const QByteArray &bytearray);
    QVariant(
const QBitArray &bitarray);
    QVariant(
const QString &string);
    QVariant(
const QLatin1String &string);
    QVariant(
const QStringList &stringlist);
    QVariant(
const QChar &qchar);
    QVariant(
const QDate &date);
    QVariant(
const QTime &time);
    QVariant(
const QDateTime &datetime);
    QVariant(
const QList<QVariant> &list);
    QVariant(
const QMap<QString,QVariant> &map);
    QVariant(
const QHash<QString,QVariant> &hash);
#ifndef QT_NO_GEOM_VARIANT
    QVariant(
const QSize &size);
    QVariant(
const QSizeF &size);
    QVariant(
const QPoint &pt);
    QVariant(
const QPointF &pt);
    QVariant(
const QLine &line);
    QVariant(
const QLineF &line);
    QVariant(
const QRect &rect);
    QVariant(
const QRectF &rect);
#endif
    QVariant(
const QUrl &url);
    QVariant(
const QLocale &locale);
#ifndef QT_NO_REGEXP
    QVariant(
const QRegExp &regExp);
#endif
#ifndef QT_BOOTSTRAPPED
    QVariant(
const QEasingCurve &easing);
#endif
    QVariant(Qt::GlobalColor color);
2.QVariant Type
在該對象中type負責記錄對象的類型,這對于正確取出對象是很用得
3.成員函數
  Type type() const;
    
int userType() const;
    
const char *typeName() const;

    
bool canConvert(Type t) const;
    
bool convert(Type t);

#ifdef QT3_SUPPORT
    inline QT3_SUPPORT 
bool canCast(Type t) const
    { 
return canConvert(t); }
    inline QT3_SUPPORT 
bool cast(Type t)
    { 
return convert(t); }
#endif
這幾個函數的用途很顯然,看看其中一個的實現吧
bool QVariant::canConvert(Type t) const
{
    
//we can treat floats as double
    
//the reason for not doing it the "proper" way is that QMetaType::Float's value is 135,
    
//which can't be handled by qCanConvertMatrix
    
//In addition QVariant::Type doesn't have a Float value, so we're using QMetaType::Float
    const uint currentType = ((d.type == QMetaType::Float) ? QVariant::Double : d.type);
    
if (uint(t) == uint(QMetaType::Float)) t = QVariant::Double;

    
if (currentType == uint(t))
        
return true;

    
if (currentType > QVariant::LastCoreType || t > QVariant::LastCoreType) {
        
switch (uint(t)) {
        
case QVariant::Int:
            
return currentType == QVariant::KeySequence
                   
|| currentType == QMetaType::ULong
                   
|| currentType == QMetaType::Long
                   
|| currentType == QMetaType::UShort
                   
|| currentType == QMetaType::UChar
                   
|| currentType == QMetaType::Char
                   
|| currentType == QMetaType::Short;
        
case QVariant::Image:
            
return currentType == QVariant::Pixmap || currentType == QVariant::Bitmap;
        
case QVariant::Pixmap:
            
return currentType == QVariant::Image || currentType == QVariant::Bitmap
                              
|| currentType == QVariant::Brush;
        
case QVariant::Bitmap:
            
return currentType == QVariant::Pixmap || currentType == QVariant::Image;
        
case QVariant::ByteArray:
            
return currentType == QVariant::Color;
        
case QVariant::String:
            
return currentType == QVariant::KeySequence || currentType == QVariant::Font
                              
|| currentType == QVariant::Color;
        
case QVariant::KeySequence:
            
return currentType == QVariant::String || currentType == QVariant::Int;
        
case QVariant::Font:
            
return currentType == QVariant::String;
        
case QVariant::Color:
            
return currentType == QVariant::String || currentType == QVariant::ByteArray
                              
|| currentType == QVariant::Brush;
        
case QVariant::Brush:
            
return currentType == QVariant::Color || currentType == QVariant::Pixmap;
        
case QMetaType::Long:
        
case QMetaType::Char:
        
case QMetaType::UChar:
        
case QMetaType::ULong:
        
case QMetaType::Short:
        
case QMetaType::UShort:
            
return qCanConvertMatrix[QVariant::Int] & (1 << currentType) || currentType == QVariant::Int;
        
default:
            
return false;
        }
    }

    
if(t == String && currentType == StringList)
        
return v_cast<QStringList>(&d)->count() == 1;
    
else
        
return qCanConvertMatrix[t] & (1 << currentType);
}
該函數作用是檢測存儲對象是否可以轉換為輸入類型,具體實現很明了
4.QVariant對象的最后一個主要的功能就是到給定類型的轉換
函數簇如下:
int toInt(bool *ok = 0const;
    
uint toUInt(bool *ok = 0const;
    qlonglong toLongLong(
bool *ok = 0const;
    qulonglong toULongLong(
bool *ok = 0const;
    
bool toBool() const;
    
double toDouble(bool *ok = 0const;
    
float toFloat(bool *ok = 0const;
    qreal toReal(
bool *ok = 0const;
    QByteArray toByteArray() 
const;
    QBitArray toBitArray() 
const;
    QString toString() 
const;
    QStringList toStringList() 
const;
    QChar toChar() 
const;
    QDate toDate() 
const;
    QTime toTime() 
const;
    QDateTime toDateTime() 
const;
    QList
<QVariant> toList() const;
    QMap
<QString, QVariant> toMap() const;
    QHash
<QString, QVariant> toHash() const;
其一個實現如下:
/*!
    \fn QTime QVariant::toTime() const

    Returns the variant as a QTime if the variant has type() \l Time,
    \l DateTime, or \l String; otherwise returns an invalid time.

    If the type() is \l String, an invalid time will be returned if
    the string cannot be parsed as a Qt::ISODate format time.

    \sa canConvert(), convert()
*/
QTime QVariant::toTime() 
const
{
    
return qVariantToHelper<QTime>(d, Time, handler);
}
使用了模板函數:qVariantToHelper
5.關于qVariantToHelper
/*!
    \fn bool QVariant::isValid() const

    Returns true if the storage type of this variant is not
    QVariant::Invalid; otherwise returns false.
*/

template 
<typename T>
inline T qVariantToHelper(
const QVariant::Private &d, QVariant::Type t,
                          
const QVariant::Handler *handler, T * = 0)
{
    
if (d.type == t)
        
return *v_cast<T>(&d);

    T ret;
    handler
->convert(&d, t, &ret, 0);
    
return ret;
}
該函數根據對象信息和目標類型做轉換工作,如果二者類型一致,則直接做轉換,否則交給函數handler->convert處理
6.關于Handler對象
    struct Handler {
        f_construct construct;
        f_clear clear;
        f_null isNull;
#ifndef QT_NO_DATASTREAM
        f_load load;
        f_save save;
#endif
        f_compare compare;
        f_convert convert;
        f_canConvert canConvert;
        f_debugStream debugStream;
    };
不過好像沒看出什么門道,那就繼續看看
其實際結構為:
const QVariant::Handler qt_kernel_variant_handler = {
    construct,
    clear,
    isNull,
#ifndef QT_NO_DATASTREAM
    
0,
    
0,
#endif
    compare,
    convert,
    
0,
#if !defined(QT_NO_DEBUG_STREAM) && !defined(Q_BROKEN_DEBUG_STREAM)
    streamDebug
#else
    
0
#endif
};
再看其中 一個函數實現constuct
static void construct(QVariant::Private *x, const void *copy)
{
    x
->is_shared = false;

    
switch (x->type) {
    
case QVariant::String:
        v_construct
<QString>(x, copy);
        
break;
    
case QVariant::Char:
        v_construct
<QChar>(x, copy);
        
break;
    
case QVariant::StringList:
        v_construct
<QStringList>(x, copy);
        
break;
    
case QVariant::Map:
        v_construct
<QVariantMap>(x, copy);
        
break;
    
case QVariant::Hash:
        v_construct
<QVariantHash>(x, copy);
        
break;
    
case QVariant::List:
        v_construct
<QVariantList>(x, copy);
        
break;
    
case QVariant::Date:
        v_construct
<QDate>(x, copy);
        
break;
    
case QVariant::Time:
        v_construct
<QTime>(x, copy);
        
break;
    
case QVariant::DateTime:
        v_construct
<QDateTime>(x, copy);
        
break;
    
case QVariant::ByteArray:
        v_construct
<QByteArray>(x, copy);
        
break;
    
case QVariant::BitArray:
        v_construct
<QBitArray>(x, copy);
        
break;
#ifndef QT_NO_GEOM_VARIANT
    
case QVariant::Size:
        v_construct
<QSize>(x, copy);
        
break;
    
case QVariant::SizeF:
        v_construct
<QSizeF>(x, copy);
        
break;
    
case QVariant::Rect:
        v_construct
<QRect>(x, copy);
        
break;
    
case QVariant::LineF:
        v_construct
<QLineF>(x, copy);
        
break;
    
case QVariant::Line:
        v_construct
<QLine>(x, copy);
        
break;
    
case QVariant::RectF:
        v_construct
<QRectF>(x, copy);
        
break;
    
case QVariant::Point:
        v_construct
<QPoint>(x, copy);
        
break;
    
case QVariant::PointF:
        v_construct
<QPointF>(x, copy);
        
break;
#endif
    
case QVariant::Url:
        v_construct
<QUrl>(x, copy);
        
break;
    
case QVariant::Locale:
        v_construct
<QLocale>(x, copy);
        
break;
#ifndef QT_NO_REGEXP
    
case QVariant::RegExp:
        v_construct
<QRegExp>(x, copy);
        
break;
#endif
#ifndef QT_BOOTSTRAPPED
    
case QVariant::EasingCurve:
        v_construct
<QEasingCurve>(x, copy);
        
break;
#endif
    
case QVariant::Int:
        x
->data.i = copy ? *static_cast<const int *>(copy) : 0;
        
break;
    
case QVariant::UInt:
        x
->data.u = copy ? *static_cast<const uint *>(copy) : 0u;
        
break;
    
case QVariant::Bool:
        x
->data.b = copy ? *static_cast<const bool *>(copy) : false;
        
break;
    
case QVariant::Double:
        x
->data.d = copy ? *static_cast<const double*>(copy) : 0.0;
        
break;
    
case QMetaType::Float:
        x
->data.f = copy ? *static_cast<const float*>(copy) : 0.0f;
        
break;
    
case QMetaType::QObjectStar:
        x
->data.o = copy ? *static_cast<QObject *const*>(copy) : 0;
        
break;
    
case QVariant::LongLong:
        x
->data.ll = copy ? *static_cast<const qlonglong *>(copy) : Q_INT64_C(0);
        
break;
    
case QVariant::ULongLong:
        x
->data.ull = copy ? *static_cast<const qulonglong *>(copy) : Q_UINT64_C(0);
        
break;
    
case QVariant::Invalid:
    
case QVariant::UserType:
        
break;
    
default:
        
void *ptr = QMetaType::construct(x->type, copy);
        
if (!ptr) {
            x
->type = QVariant::Invalid;
        } 
else {
            x
->is_shared = true;
            x
->data.shared = new QVariant::PrivateShared(ptr);
        }
        
break;
    }
    x
->is_null = !copy;
}
繼續看v_construct
該函數模板實現和平臺有關,其中一個平臺的實現如下:
template <class T>
inline 
void v_construct(QVariant::Private *x, const void *copy, T * = 0)
{
    
if (sizeof(T) > sizeof(QVariant::Private::Data)) {
        x
->data.shared = copy ? new QVariantPrivateSharedEx<T>(*static_cast<const T *>(copy))
                              : 
new QVariantPrivateSharedEx<T>;
        x
->is_shared = true;
    } 
else {
        
if (copy)
            
new (&x->data.ptr) T(*static_cast<const T *>(copy));
        
else
            
new (&x->data.ptr) T;
    }
}
這里主要是把傳入對象指針導入為本身的數據指針data.ptr

QVariant大致就這個樣子
posted on 2011-12-10 20:51 ccsdu2009 閱讀(4378) 評論(0)  編輯 收藏 引用 所屬分類: QT源碼分析
 
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            欧美成人午夜视频| 性欧美8khd高清极品| 久久综合给合久久狠狠狠97色69| 午夜精品国产| 樱花yy私人影院亚洲| 欧美大色视频| 欧美黄色一区二区| 亚洲天堂av高清| 在线视频精品一区| 国产综合欧美在线看| 久久日韩粉嫩一区二区三区 | 亚洲欧美999| 午夜宅男欧美| 亚洲欧洲在线一区| 亚洲美女视频在线观看| 国产日韩欧美成人| 欧美护士18xxxxhd| 国产精品白丝av嫩草影院 | 亚洲高清资源| 99国产一区二区三精品乱码| 国产美女精品人人做人人爽| 欧美大片在线观看一区二区| 欧美视频在线一区| 蜜桃久久av| 国产精品狠色婷| 美女视频黄a大片欧美| 欧美日韩久久| 久久久久久网站| 欧美日韩一区在线观看视频| 看片网站欧美日韩| 国产精品a久久久久| 欧美成人免费在线| 国产精品一区二区你懂得| 免费成人av资源网| 国产精品日韩精品欧美精品| 亚洲第一精品影视| 国产精品美女视频网站| 亚洲国产精品福利| 狠狠色噜噜狠狠狠狠色吗综合| 亚洲精品久久久久久久久久久| 国产一区视频观看| 亚洲免费人成在线视频观看| 一区二区欧美在线| 久久综合伊人77777麻豆| 欧美自拍偷拍午夜视频| 欧美精品videossex性护士| 麻豆久久婷婷| 黑人操亚洲美女惩罚| 亚洲视频综合在线| 一区二区三区欧美日韩| 美女成人午夜| 欧美成人一品| 免费成人高清视频| 精品不卡一区| 久久精品国产精品亚洲| 久久精品国产99精品国产亚洲性色| 欧美三级中文字幕在线观看| 亚洲精品国产日韩| 亚洲乱码国产乱码精品精天堂| 葵司免费一区二区三区四区五区| 久久久久久综合网天天| 国内久久视频| 久久精品女人| 久久影音先锋| 亚洲国产高清在线| 蜜桃久久av| 亚洲人成亚洲人成在线观看| 一本色道综合亚洲| 欧美午夜片欧美片在线观看| 中文一区字幕| 午夜欧美精品| 国产综合欧美| 欧美二区在线看| 日韩视频一区二区三区在线播放免费观看| 日韩亚洲一区二区| 国产精品扒开腿爽爽爽视频| 亚洲在线免费| 老牛嫩草一区二区三区日本| 亚洲国产成人午夜在线一区| 欧美精品一线| 亚洲一区视频在线| 久久五月天婷婷| 亚洲毛片一区| 国产精品乱子久久久久| 久久精品国产99国产精品| 欧美第一黄色网| 一区二区国产日产| 国产婷婷一区二区| 美女主播一区| 999亚洲国产精| 久久久蜜桃一区二区人| 91久久精品日日躁夜夜躁欧美 | 欧美日韩在线直播| 欧美在线精品一区| 亚洲国产精品一区二区第四页av| 99国产成+人+综合+亚洲欧美| 欧美日韩一区高清| 久久国产一区二区三区| 亚洲国产精品久久久久秋霞影院| 亚洲新中文字幕| **性色生活片久久毛片| 欧美视频在线免费看| 久久在线精品| 亚洲影院在线| 亚洲激情视频在线观看| 久久成人免费日本黄色| 亚洲精选成人| 精品成人国产| 国产精品一区二区久久精品| 欧美—级a级欧美特级ar全黄| 性欧美大战久久久久久久免费观看| 一区二区视频欧美| 久久五月婷婷丁香社区| 亚洲美洲欧洲综合国产一区| 欧美在线播放一区二区| 亚洲人成网站在线观看播放| 国产精品永久入口久久久| **性色生活片久久毛片| 国产精品综合视频| 欧美三级黄美女| 久久综合成人精品亚洲另类欧美| 午夜精品久久99蜜桃的功能介绍| 亚洲精品视频一区二区三区| 美日韩丰满少妇在线观看| 一本久久青青| 99精品免费网| 亚洲精品乱码久久久久久蜜桃麻豆| 国产亚洲精品bt天堂精选| 欧美午夜在线观看| 欧美日韩亚洲一区三区| 欧美国产欧美亚洲国产日韩mv天天看完整| 香蕉久久精品日日躁夜夜躁| 亚洲午夜视频| 亚洲天堂成人| 亚洲一二三区视频在线观看| 99精品欧美一区二区蜜桃免费| 亚洲高清不卡在线| 国产精品一区二区三区免费观看| 欧美精品在线观看| 国产精品多人| 欧美日本高清一区| 欧美日本不卡视频| 欧美亚州韩日在线看免费版国语版| 欧美日韩一二三区| 欧美色一级片| 国产精品五月天| 国产日产欧产精品推荐色| 国产日产高清欧美一区二区三区| 国产精品一区二区你懂得| 国产丝袜一区二区三区| 韩国视频理论视频久久| 在线观看视频一区二区欧美日韩| 亚洲动漫精品| 亚洲免费观看| 亚洲永久免费| 久久精品夜色噜噜亚洲a∨ | 伊人成人在线| 亚洲国产欧美一区二区三区丁香婷| 亚洲国产成人av| 一本色道久久加勒比精品| 亚洲欧美日本伦理| 久久国产视频网站| 欧美成人精品高清在线播放| 亚洲激情第一页| 亚洲视频在线一区观看| 久久精品99无色码中文字幕| 老色批av在线精品| 欧美日韩三级视频| 国产午夜精品久久久| 在线国产亚洲欧美| 亚洲视频一区二区在线观看| 久久av最新网址| 亚洲国产欧美不卡在线观看| 亚洲天堂网在线观看| 久久久久www| 欧美视频一区| 尤物视频一区二区| 亚洲已满18点击进入久久| 老司机精品导航| 在线亚洲一区观看| 久久精品人人做人人综合| 欧美日韩免费看| 在线观看亚洲| 午夜精品影院| 亚洲精品欧美极品| 久久久91精品国产一区二区三区 | 亚洲欧美日韩在线播放| 男人的天堂成人在线| 国产美女扒开尿口久久久| 日韩视频免费看| 久久成人免费网| 99精品国产一区二区青青牛奶| 久久五月激情| 国产亚洲精品一区二区| 亚洲午夜一区二区| 亚洲高清成人| 久久亚洲影音av资源网| 国产亚洲视频在线| 亚洲摸下面视频| 99一区二区|