- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在C++中实现一种超模块化体系结构的基本反射(reflection),该构架实际上将所有功能都作为插件加载并在运行时动态解释。由于系统具有结构自组织的独特元素,因此组件需要一些相互检查的方法(例如,需要这种反射风格的情况的示例,请参见此问题:“Best fit” dynamic type matching for plugins in C++)。
到目前为止,该体系结构是用C#开发的,但我现在正在研究如何用C++来实现。此时,我已基于以下模型为“穷人的倒影”创建了骨架:
一个Type
类,用于保存相关的类信息:
namespace Reflection {
class Type {
public:
Type(Object &, string, bool(*)(Type *));
~Type();
Object & RefObj();
string Name();
bool IsAssignableFrom(Type *);
private:
Object & _refObj;
string _name;
bool(*_isAssignableFrom_Handler)(Type *);
};
}
Object
类,反射模型中的所有参与者都将从该类下降:
class Object {
public:
Object();
virtual ~Object();
virtual string ToString();
virtual Reflection::Type * GetType();
static Reflection::Type * Type();
static bool IsAssignableFrom(Reflection::Type *);
private:
static Object _refObj;
static Reflection::Type _type;
};
string Object::ToString() { return GetType()->Name(); }
// all derived classes must implement the equivalent of this:
Reflection::Type * Object::GetType() { return &_type; }
Object Object::_refObj;
Reflection::Type Object::_type(_refObj, "Object", Object::IsAssignableFrom);
Reflection::Type * Object::Type() { return &_type; }
bool Object::IsAssignableFrom(Reflection::Type * type) {
return dynamic_cast<Object*>(&type->RefObj()) != nullptr;
}
Object
)中进行反射即可。因此,以上代码现在使我能够:
instance.GetType()
class::Type()
(instance.GetType() == class::Type())
或(instanceA.GetType() == instanceB.GetType())
is
和推论继承关系的关键:(instanceA.GetType()->IsAssignableFrom(instanceB.GetType()))
Activator.CreateInstance
类似)。但是与“适当的”反射(reflect)不同,后者本质上是一种自上而下的方法,其中有关类的[元]信息是在编译器级别收集/集中管理的,它是手动进行的,并且是自下而上的,将知识分发到对象本身中并给出他们在运行时彼此之间进行通信的一种方式。因此,要使其工作,此系统中包含的每个类都需要实现与
Object
类相同的成员和功能,以封装自身的相关方面以进行“导出”。举个例子,
Plugin
类看起来像这样(定义):
Reflection::Type * Plugin::GetType() { return &_type; }
Plugin Plugin::_refObj;
Reflection::Type Plugin::_type(_refObj, "Plugin", Plugin::IsAssignableFrom);
Reflection::Type * Plugin::Type() { return &_type; }
bool Plugin::IsAssignableFrom(Reflection::Type * type) {
return dynamic_cast<Plugin*>(&type->RefObj()) != nullptr;
}
Object
相同。几乎所有这些功能仅因其类类型(和名称)而异。
REFLECTION_H(TYPE_NAME)
,并在类定义中添加
REFLECTION_CPP(TYPE_NAME)
,所有反射样板都会自动包含在内。然后,我可以像往常一样添加特定于类的成员,而不必考虑反射,因为知道所需的所有管道均已就绪并且一切正常。例如,在当前的实现中,新的
Surface
类现在看起来像这样:
class Surface : public Plugin
{
REFLECTION_H(Surface)
public:
// ...class specific public member declarations...
private:
// ...class specific private member declarations...
};
REFLECTION_CPP(Surface);
Surface::~Surface() {}
// ...class specific member definitions...
#define REFLECTION_H(TYPE_NAME) \
public:\
virtual ~TYPE_NAME();\
static Reflection::Type& Type();\
private:\
virtual Reflection::Type& _getType();\
static TYPE_NAME _refObj;\
static Reflection::Type _type;\
static bool IsAssignableFrom(Reflection::Type&);\
static plugin_ptr CreateInstance();
#define REFLECTION_CPP(TYPE_NAME) \
Reflection::Type& TYPE_NAME::Type() { return _type; }\
Reflection::Type& TYPE_NAME::_getType() { return _type; }\
TYPE_NAME TYPE_NAME::_refObj;\
Reflection::Type TYPE_NAME::_type(_refObj, #TYPE_NAME, true, IsAssignableFrom, CreateInstance);\
bool TYPE_NAME::IsAssignableFrom(Reflection::Type& type) { return dynamic_cast<TYPE_NAME*>(&type.RefObj()) != nullptr; }\
plugin_ptr TYPE_NAME::CreateInstance() { return plugin_ptr(new TYPE_NAME); }
最佳答案
Python
由于您提到自己在Visual Studio中,因此我在下面编写的宏可能对您不起作用(根据我的经验,宏可能是令人讨厌的跨平台)。因此,这是一个Python示例,您可以将其作为预构建脚本运行,以基于文本文件中的名称生成* .cpp文件。
create_classes.py
template = """Reflection::Type * {0}::GetType() {{
return &_type;
}}
// static type info
{0} {0}::_refObj;
Reflection::Type {0}::_type(_refObj, "{0}", {0}::IsAssignableFrom);
Reflection::Type * {0}::Type() {{
return &_type;
}}
bool {0}::IsAssignableFrom(Reflection::Type * type) {{
return dynamic_cast<{0}*>(&type->RefObj()) != nullptr;
}}
"""
if __name__ == '__main__':
with open('classes', 'r') as classes:
for class_name in classes:
class_name = class_name.strip()
with open('{0}.cpp'.format(class_name), 'w') as source:
source.write(template.format(class_name))
类(文本文件)
Blossom
Bubbles
Buttercup
它将使用上面的模板创建Blossom.cpp,Bubbles.cpp和Buttercup.cpp。您可以自行决定将正确的名称添加到“类”文本文件中。 :)
#define GET_TYPE_METHOD(X) \
Reflection::Type * X::GetType() { return &_type; }
#define GET_REF_OBJ(X) X X::_refObj;
#define GET_UNDERSCORE_TYPE(X) \
Reflection::Type X::_type(_refObj, #X, X::IsAssignableFrom);
#define GET_TYPE(X) \
Reflection::Type * X::Type() { return &_type; }
#define GET_IS_ASSIGNABLE_FROM(X) \
bool X::IsAssignableFrom(Reflection::Type * type) { return dynamic_cast<X*>(&type->RefObj()) != nullptr; }
GET_TYPE_METHOD(Keeler)
GET_REF_OBJ(Keeler)
GET_UNDERSCORE_TYPE(Keeler)
GET_TYPE(Keeler)
GET_IS_ASSIGNABLE_FROM(Keeler)
如果运行
g++ -E macros.cpp
,则将获得预处理器输出。查看预处理器的想法:
$ g++ -E macros.cpp
# 1 "macros.cpp"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "macros.cpp"
# 16 "macros.cpp"
Reflection::Type * Keeler::GetType() { return &_type; }
Keeler Keeler::_refObj;
Reflection::Type Keeler::_type(_refObj, "Keeler", Keeler::IsAssignableFrom);
Reflection::Type * Keeler::Type() { return &_type; }
bool Keeler::IsAssignableFrom(Reflection::Type * type) { return dynamic_cast<Keeler*>(&type->RefObj()) != nullptr; }
这是否符合您的需求?
关于c++ - C++中的“Poor Man'反射”(又名自下而上的反射),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21057938/
一、反射 1.定义 Java的反射(reflection)机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法(即使是私有的);对于任意一个对象,都能够调用它的任意方法和属性,那么,我
有没有办法从 JavaScript 对象内部获取所有方法(私有(private)、特权或公共(public))?这是示例对象: var Test = function() { // private m
我有一个抽象类“A”,类“B”和“C”扩展了 A。我想在运行时根据某些变量创建这些实例。如下所示: public abstract class A { public abstract int
假设我们在内存中有很多对象。每个都有一个不同的ID。如何迭代内存以找到与某些 id 进行比较的特定对象?为了通过 getattr 获取并使用它? 最佳答案 您应该维护这些对象的集合,因为它们是在类属性
假设我有这个结构和一个方法: package main import ( "fmt" "reflect" ) type MyStruct struct { } func (a *MyS
C#反射简介 反射(Reflection)是C#语言中一种非常有用的机制,它可以在运行时动态获取对象的类型信息并且进行相应的操作。 反射是一种在.NET Framework中广
概述 反射(Reflection)机制是指在运行时动态地获取类的信息以及操作类的成员(字段、方法、构造函数等)的能力。通过反射,我们可以在编译时期未知具体类型的情况下,通过运行时的动态
先来看一段魔法吧 public class Test { private static void changeStrValue(String str, char[] value) {
结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套; go中的struct类型理解为类,可以定义方法,和函数定义有些许区别; struct类型是值类型
反射 1. 反射的定义 Java的反射(reflection)机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到那么,我们
反射的定义 java的反射(reflection) 机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性,既然能拿到嘛,那么,我们就可以
我有一个 Java POJO: public class Event { private String id; private String name; private Lon
我编写了以下函数来检查给定的单例类是否实现了特征。 /** Given a singleton class, returns singleton object if cls implements T.
我正在研究 Java 反射的基础知识并观察有关类方法的信息。我需要获得一个符合 getMethod() 函数描述的规范的方法。然而,当我这样做时,我得到了一个 NoSuchMethodExceptio
我正在通过以下代码检索 IEnumerable 属性列表: BindingFlags bindingFlag = BindingFlags.Instance | BindingFlags.Public
我需要检查属性是否在其伙伴类中定义了特定属性: [MetadataType(typeof(Metadata))] public sealed partial class Address { p
我正在尝试使用 Reflections(由 org.reflections 提供)来处理一些繁重的工作,因此我不需要在很长的时间内为每个类手动创建一个实例列表。但是,Reflections 并未按照我
scala 反射 API (2.10) 是否提供更简单的方法来搜索加载的类并将列表过滤到实现定义特征的特定类? IE; trait Widget { def turn(): Int } class
我想在运行时使用反射来查找具有给定注释的所有类,但是我不知道如何在 Scala 中这样做。然后我想获取注释的值并动态实例化每个映射到关联注释值的带注释类的实例。 这是我想要做的: package pr
这超出了我的头脑,有人可以更好地向我解释吗? http://mathworld.wolfram.com/Reflection.html 我正在制作一个 2d 突破格斗游戏,所以我需要球能够在它击中墙壁
我是一名优秀的程序员,十分优秀!