由于QMetaObject中存储了QObject子类对应的方法、属性等信息,本篇着重讲解QMetaMethod中的方法及QMetaObject中的匿名结构体中data、stringdata的对应关系;
为了更好的说明QMetaObject中匿名结构体的作用,特举例说明
ReflectDemo继承自QObject,定义了2个方法,一个成员变量,成员方法使用Q_INVOKABLE声明(如果不加Q_INVOKABLE关键字,QMetaObject中将不会存储对应的方法,既无法进行反射;信号、槽等不需要),Q_CLASSINFO声明类的相关信息,针对以上这个简单的类,QMetaObject中将按照如下图所示进行存储;
data中0~13的位置之前已说明主要是存储类名、信息及方法数量、方法在data中的起始位置、property数量、property在data中的起始位置、枚举的数量及在data中的起始位置等信息;
ReflectDemo中由于只有一处声明了Q_CLASSINFO,所以classInfoCount为1,其起始位置是从14开始,所以classInfoData为14;data[14]的值代表了其在stringdata中的索引下表,所以要访问clssInfo的name时,需要先取出data[14]的值,在取出其在stringdata中的字符串,即:name = stringdata[data[14]];data[15]存储的是classInfo中name对应的value值,其访问方式为stringdata[data[15]];
由于ReflectDemo中classInfo的数量为1,所以Method的起始位置为:14 + classInfoCount * 2,此例即为16;
data[16]存储的是ReflectDemo中使用Q_INVOKABLE声明的方法(或者标记了信号、槽;按照顺序存储)setName在stringdata中的索引,通过stringdata[data[16]]获取到对应方法的名称;
data[17]存储的是setName方法的参数数量;
data[18]存储的是setName方法返回值的类型及参数类型、名称的索引位置;
data[19]存储的是setName方法的tag在stringdata中的偏移量,即stringdata[data[19]];
data[20]存储的是setName方法的可访问属性及当前方法表示的是信号或者槽或者是普通方法;
data[21]存储的是下一个方法的偏移量,data[21]~data[25]存储的内容和data[16]~data[20]是一样的,此处不再解释;
data[26]存储的是setName方法返回的类型,其访问方式是通过QMetaType将类型id进行转换;
data[27]存储的是setName中第一个参数的类型;
data[28]存储的是setName中第二个参数的类型;
data[29]存储的是setName中第一个参数的名称在stringdata中的索引位置,此处为5
data[30]存储的是setName中第二个参数的名称在stringdata中的索引位置,此处为6;
data[31]存储的是getName中返回值的类型;
综上,property的偏移量及propertyData的值为32;
propertyData = metaMethodData + 5 * metaMethodCount + 每个方法的返回值1 + 每个方法的参数 * 2;
搞清楚了上图的关系,在看QMetaMethod中的方法时就简单很多:
如获取方法名称:
handle的值即为每个方法在data中的偏移量,先从data中取出其在stringdata中的偏移量,在从stringdata中取出对应的字符串
获取参数数量:
QMetaMethod中的普通访问方法,全都是操作data及stringdata来获取的,不在赘述;