我正在制作一个简单的消息传递实体系统。我有一个连接到工厂的实体描述符表,用于在运行时创建实体子类,我希望拥有它以便可以通过字符串创建它们:
EntityManager manager; //managers have all of the entity table information (See below)
//Counter is a sample class that inherits from Entity
Counter* counter = manager.makeEntity("Counter"); //the string doesn't have to match the class name.
现在,我知道我可以使用一个简单的 switch 语句,但我希望系统也可以扩展——也就是说,当我(或我系统的其他用户)想要创建一个新的 Entity 子类时,我不必须转到开关 block 并添加它。目前,我使用宏来创建我静态实例化的辅助类,以便构造函数将条目添加到实体表中。这些类还初始化实体并从构造函数中删除大量样板文件。
//EHandle is a wrapper for Entity*. Currently std::shared_ptr<Entity>
class GenericDesc
{
public:
virtual ~GenericDesc() {}
virtual EHandle makeEntity() const =0;
};
namespace Descriptor
{
//Adds a descriptor to an internal map<string, GenericDesc*>
void addEntityDescriptor(const std::string& type, GenericDesc& desc);
EHandle newEntity(const std::string& type); //Factory method
}
//Add this to every entity class definition
#define DECLARE_ENTITY_CLASS(CLASS_NAME) \
friend class CLASS_NAME##Descriptor;
//Use these after a class definition to add the entity class to the descriptor table
#define BEGIN_ENTITY_TYPE(ENTITY_NAME, CLASS_NAME, BASE_NAME) \
BEGIN_ENTITY_TYPE_GUTS(ENTITY_NAME, CLASS_NAME) \
BASE_NAME##Descriptor::prepareEntity(ent);
#define BEGIN_ENTITY_TYPE_BASELESS(ENTITY_NAME, CLASS_NAME) \
BEGIN_ENTITY_TYPE_GUTS(ENTITY_NAME, CLASS_NAME) \
ent->self = ent;
#define BEGIN_ENTITY_TYPE_GUTS(ENTITY_NAME, CLASS_NAME) \
class CLASS_NAME##Descriptor : public GenericDesc \
{ \
private: \
typedef CLASS_NAME ClassName; \
public: \
CLASS_NAME##Descriptor() \
{ \
Descriptor::addEntityDescriptor(ENTITY_NAME, *this); \
} \
virtual ~CLASS_NAME##Descriptor() {} \
virtual EHandle makeEntity() const\
{ \
auto ent = std::shared_ptr<CLASS_NAME>(new CLASS_NAME); \
prepareEntity(ent); \
ent->type = ENTITY_NAME; \
return ent; \
} \
static void prepareEntity(std::shared_ptr<ClassName> ent) \
{
//These functions are caled between BEGIN_ENTITY_TYPE and END_ENTITY_TYPE
//ADD_ENTITY_INPUT binds a function to a string
#define ADD_ENTITY_INPUT(INPUT_NAME, INPUT_FUNC) \
ent->addInput(INPUT_NAME, std::bind(&ClassName::INPUT_FUNC, ent, std::placeholders::_1));
//ADD_ENTITY_OUTPUT binds an Output object to a string
#define ADD_ENTITY_OUTPUT(OUTPUT_NAME, OUTPUT_OBJECT) \
ent->addOutput(OUTPUT_NAME, ent->OUTPUT_OBJECT);
#define END_ENTITY_TYPE(CLASS_NAME) \
} \
}; \
static CLASS_NAME##Descriptor CLASS_NAME##Desc; //TODO: find a way to fix the multiple-static-allocation issue
想法是您创建一个 BEGIN_ENTITY_TYPE(...) END_ENTITY_TYPE(...) 子句,中间有 ADD_ENTITY_x 位。我的问题是,是否有一种不那么宏大的方法可以做到这一点,它仍然可以最大限度地减少样板文件,并且不需要修改定义实体子类的文件之外的任何文件。模板类可以工作,但我不知道如何使用模板类执行 ADD_ENTITY_INPUT/OUTPUT 操作。
最佳答案
这可能不完全是您所追求的,但请考虑以下内容:
class Factory
{
public:
virtual void InitialiseFactory() = 0;
Entity* CreateEntity( string type )
{
// (obviously, you must handle the string not being found.)
return m_MapEntities[ type ]->Clone();
}
};
class MyFactory : public Factory
{
public:
void InitialiseFactory()
{
m_MapEntities["typea"] = new MyEntity();
}
};
这里的想法是查找在基类中具体化,基类可以保持不变,但是提供哪些实体的细节在派生类的实现中。
另一种方法是在每个实体中调用要添加的工厂的静态方法,但仍然需要从某个地方调用该方法。
关于c++ - 使类信息在运行时可访问的可扩展方式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9470832/
总的来说,我对ruby还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用
我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0
大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje
我试图获取一个长度在1到10之间的字符串,并输出将字符串分解为大小为1、2或3的连续子字符串的所有可能方式。例如:输入:123456将整数分割成单个字符,然后继续查找组合。该代码将返回以下所有数组。[1,2,3,4,5,6][12,3,4,5,6][1,23,4,5,6][1,2,34,5,6][1,2,3,45,6][1,2,3,4,56][12,34,5,6][12,3,45,6][12,3,4,56][1,23,45,6][1,2,34,56][1,23,4,56][12,34,56][123,4,5,6][1,234,5,6][1,2,345,6][1,2,3,456][123
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
exe应该在我打开页面时运行。异步进程需要运行。有什么方法可以在ruby中使用两个参数异步运行exe吗?我已经尝试过ruby命令-system()、exec()但它正在等待过程完成。我需要用参数启动exe,无需等待进程完成是否有任何rubygems会支持我的问题? 最佳答案 您可以使用Process.spawn和Process.wait2:pid=Process.spawn'your.exe','--option'#Later...pid,status=Process.wait2pid您的程序将作为解释器的子进程执行。除
我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r
我的瘦服务器配置了nginx,我的ROR应用程序正在它们上运行。在我发布代码更新时运行thinrestart会给我的应用程序带来一些停机时间。我试图弄清楚如何优雅地重启正在运行的Thin实例,但找不到好的解决方案。有没有人能做到这一点? 最佳答案 #Restartjustthethinserverdescribedbythatconfigsudothin-C/etc/thin/mysite.ymlrestartNginx将继续运行并代理请求。如果您将Nginx设置为使用多个上游服务器,例如server{listen80;server