本文述说了动态反射库Ponder中的组织结构。
所有的动态反射库中,总是有一种能够存储任意类型的类(用来对标std::any
)。meta
中就叫any
,而Ponder
中是Value
。
这种any
类最简单的实现通常包含两个数据:
1
2
3
4
5
6
7
| class any {
public:
...
private:
void* data; // 用于存储擦除了类型的数据
TypeInfo typeinfo; // 用于存储此数据对应的类型信息
};
|
而在Ponder
中的实现则更加复杂:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
| // include/ponder/value.hpp 72
class PONDER_API Value
{
public:
... 一些操作函数,不是重点,忽略
static const Value nothing;
private:
typedef mapbox::util::variant<
NoType, bool, long, double, ponder::String, EnumObject, UserObject
> Variant;
Variant m_value; // Stored value
ValueKind m_type; // Ponder type of the value
};
|
Variant
是对标std::variant
的类型,存储了Value
所有可能的类型
ValueKind
则是一个枚举,用来告知m_value
目前代表什么类型:
1
2
3
4
5
6
7
8
9
10
11
12
| // include/ponder/type.hpp 65
enum class ValueKind
{
None, ///< No type has been defined yet
Boolean, ///< Boolean type (`bool`)
Integer, ///< Integer types (`unsigned`,`signed` `char` `short` `int` `long`)
Real, ///< Real types (`float`, `double`)
String, ///< String types (`char*`, `ponder::String`)
Enum, ///< Enumerated types
Array, ///< Array types (`T[]`, `std::vector`, `std::list`)
User ///< User-defined classes
};
|
注意到Value
中Variant<...>
的模板参数,除了布尔,整数(统一用long
存储),浮点数(统一用double
存储)之外,还有三种自定义类型:
ponder::String
:用于存储字符串类型,其具体类型由用户通过定义宏自己选择(std::string
或string_view
),具体实现见include/ponder/detail/idtraits.hpp
。这里的string_view
是作者自己实现的,因为Ponder
的标准是C++14
。EnumObject
:对枚举类型特化的类,用于存储枚举值UserObject
:对类类型特化的类,用于存储类对象
EnumObject
是很简单的结构,用于存储枚举值和其名称,可以获得其值和名称:
1
2
3
4
5
6
7
8
9
10
11
12
| // include/ponder/enumobject.hpp
class PONDER_API EnumObject
{
public:
... 一些操作,不细说了
private:
long m_value; /// 此枚举量的值
const Enum* m_enum; ///< 此枚举值对应的枚举类型信息
};
|
Enum
则是存储一个枚举类型信息的类,主要成员为:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
| // include/ponder/enum.hpp
class PONDER_API Enum : public Type
{
PONDER__NON_COPYABLE(Enum); // 这个宏指定此类不可被拷贝,其实是禁用了类的拷贝&复制运算符
typedef long EnumValue; // 要存储的枚举值的类型
... 一些操作,不细说了
private:
Enum(IdRef name); // 构造函数,给入这个枚举类型的名称
typedef detail::Dictionary<Id, IdRef, EnumValue> EnumTable;
Id m_name; ///< 枚举类型的名称
EnumTable m_enums; ///< 此枚举类型里面的枚举值们
};
|
所有的枚举值使用EnumTable
存储,这是作者自己造的一个哈希表,文档中说对Cache更友好。里面存储着枚举值的名称(默认是字符串,也是可以通过宏更改)和枚举值。
UserObject
的结构也很简单:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
| // include/ponder/userobject.hpp
class PONDER_API UserObject
{
public:
... 一些接口,不细说了
private:
friend class Property;
// 将一个值设置给类的属性
void set(const Property& property, const Value& value) const;
UserObject(const Class* cls, detail::AbstractObjectHolder* h)
: m_class(cls)
, m_holder(h)
{}
/// 类的类型信息
const Class* m_class;
/// 类对象的存储类
std::shared_ptr<detail::AbstractObjectHolder> m_holder;
};
|
这里Class
存储类的类型信息。detaul::AbstractObjectHolder
则是抽象类,用于存储一个类对象。
Class
的定义如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
| // include/ponder/class.hpp
class PONDER_API Class : public Type
{
PONDER__NON_COPYABLE(Class);
// Structure holding informations about a base metaclass
struct BaseInfo
{
const Class* base;
int offset;
};
// These are shared_ptr as the objects can be inherited. When this happens the
// pointers are copied.
typedef std::shared_ptr<Constructor> ConstructorPtr;
typedef std::shared_ptr<Property> PropertyPtr;
typedef std::shared_ptr<Function> FunctionPtr;
typedef std::vector<BaseInfo> BaseList;
typedef std::vector<ConstructorPtr> ConstructorList;
typedef detail::Dictionary<Id, IdRef, PropertyPtr> PropertyTable;
typedef detail::Dictionary<Id, IdRef, FunctionPtr> FunctionTable;
typedef void (*Destructor)(const UserObject&, bool);
typedef UserObject (*UserObjectCreator)(void*);
std::size_t m_sizeof; // Size of the class in bytes.
Id m_id; // Name of the metaclass
FunctionTable m_functions; // Table of metafunctions indexed by ID
PropertyTable m_properties; // Table of metaproperties indexed by ID
BaseList m_bases; // List of base metaclasses
ConstructorList m_constructors; // List of metaconstructors
Destructor m_destructor; // Destructor (function able to delete an abstract object)
UserObjectCreator m_userObjectCreator; // Convert pointer of class instance to UserObject
... 一些操作,不细说了
};
|
Class
中存储着:
m_sizeof
:此类的大小m_id
:此类的ID(一般是字符串,也就是此类的名称)m_functions
:此类中的函数,使用作者自制的哈希表存储m_properties
:此类中的成员变量,使用作者自制的哈希表存储m_bases
:父类信息m_constructors
:构造函数信息m_destructor
:析构函数信息m_userObjectCreator
:
Property
是存储类中成员变量信息的类:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
| class PONDER_API Property : public Type
{
PONDER__NON_COPYABLE(Property);
public:
virtual ~Property();
IdReturn name() const;
ValueKind kind() const;
virtual bool isReadable() const;
virtual bool isWritable() const;
Value get(const UserObject& object) const;
void set(const UserObject& object, const Value& value) const;
virtual void accept(ClassVisitor& visitor) const;
protected:
template <typename T> friend class ClassBuilder;
friend class UserObject;
Property(IdRef name, ValueKind type);
virtual Value getValue(const UserObject& object) const = 0;
virtual void setValue(const UserObject& object, const Value& value) const = 0;
private:
Id m_name; // Name of the property
ValueKind m_type; // Type of the property
};
|
存储的是成员变量的名称和类型。我特意将set()
和get()
函数的声明留下,这说明这个类是可以设置/获得成员变量值的类型的。但是他本身是没有存储变量名称的,所以有一些函数