c++ - 很难确定 std::map 的正确值类型
问题描述
附录:
让我简化我的问题,一次尝试这一步,因为我觉得我对我想要实现的目标造成了太多的困惑。
请注意,我使用的是 C++11,并且我认为在使用 PHP-CPP 库时我一定会使用它。但是,由于我是一个完全的初学者,我什至不确定这一点。
我的最终目标是“简单地”创建一个可以保存模板返回的任何值的映射。Php::Class
但是,让我们从尝试一个具体的开始,例如Php::Class<Animal::Mammal>
。
首先,我在命名空间中创建了一个Animal
名为的类Mammal
,它的扩展自Php::Base
(当你想用 PHP-CPP 将一个类导出到 PHP 时,它需要扩展Php::Base
):
namespace Animal {
class Mammal : public Php::Base { /* left out for brevity */ }
}
然后,为了简单起见,让我现在尝试在内部创建一个地图get_module()
,然后尝试在其中添加一个元素:
extern "C"
{
PHPCPP_EXPORT void *get_module() {
// create the map
std::map<std::string, Php::Class<Animal::Mammal>> classMap;
// create the thing (at this point I just don't know what the actual return type is anymore) that will be exported later on.
Php::Class<Animal::Mammal> mammal( "Mammal" );
// try to add this "thing" to the map:
classMap[ "Animal::Mammal" ] = mammal;
/* left out for brevity */
}
这导致:
error: use of deleted function ‘Php::Class<Animal::Mammal>&
Php::Class<Animal::Mammal>::operator=(const Php::Class<Animal::Mammal>&)’
当我尝试:
// require a pointer
std::map<std::string, Php::Class<Animal::Mammal>*> classMap;
classMap[ "Animal::Mammal" ] = *mammal;
这导致:
error: no match for ‘operator*’ (operand type is ‘Php::Class<Animal::Mammal>’)
我也尝试了许多其他的东西,但似乎没有什么可以让我将模板Php::Class
创建的任何内容放入地图中。在这一点上,我正在拔头发。我错过了什么?
是不是模板Php::Class
创建了一些根本无法放入地图的东西?我如何找出模板Php::Class
实际创建的东西是什么?
原始问题:
尽管我对 C++ 有一些粗略的了解,但我对实际使用它进行编程还是很陌生。
我正在尝试使用PHP-CPP开发 PHP 扩展。使用这个库,您可以定义 C++ 类并将它们导出为 PHP 类。他们的文档几乎只展示了在其主函数中导出这些类和方法的示例。get_module()
然而,由于这很快就会变得相当混乱,我想把这个逻辑移到靠近各个类的地方。所以,我为此创建了一个模板函数:
namespace Animal
{
template<typename T>
void addToExtension( Php::Extension &extension, Php::Namespace &ns, ClassMap &map ) { }
}
然后我专门研究以下内容:
namespace Animal
{
class Mammal : public Php::Base
{
};
template<>
inline void addToExtension<Mammal>( Php::Extension &extension, Php::Namespace &ns, ClassMap &map ) {
Php::Class<Mammal> clss( "Mammal" );
// this is where I'm having extreme difficulty with
map[ "Animal::Mammal" ] = clss;
// add class to namespace
ns.add( std::move( clss ) );
}
}
...然后get_module()
我这样做:
PHPCPP_EXPORT void *get_module() {
static Php::Extension extension( "animals", "0.1" );
Php::Namespace ns( "Animal" );
Animal::ClassMap classMap;
Animal::addToExtension<Animal::Mammal>( extension, ns, classMap );
Animal::addToExtension<Animal::Dog>( extension, ns, classMap );
Animal::addToExtension<Animal::Cat>( extension, ns, classMap );
etc...
extension.add( std::move( ns ) );
return extension;
}
如您所见,我还想将每个单独的类添加到Animal::ClassMap
地图中,因为我想在后续类需要从它们扩展时使用它们(即导出到 PHP):
// for instance, something like (not sure yet, if this is actually possible)
clss.extends( map[ "Animal::Mammal" ] );
但这就是我遇到的极大困难。我试图以Animal::ClassMap
各种方式定义:
namespace Animal
{
// some examples of what I've tried:
typedef std::map<std::string, Php::Class> ClassMap;
typedef std::map<std::string, Php::Class*> ClassMap;
typedef std::map<std::string, Php::Base> ClassMap;
typedef std::map<std::string, Php::Base*> ClassMap;
typedef std::map<std::string, Php::ClassBase> ClassMap;
typedef std::map<std::string, Php::ClassBase*> ClassMap;
typedef std::map<std::string, Php::Class<Php::Base>> ClassMap;
typedef std::map<std::string, Php::Class<Php::Base>*> ClassMap;
typedef std::map<std::string, std::shared_ptr<Php::Class>> ClassMap;
// etc., etc., etc. ...
}
并试图以各种方式分配给它:
Php::Class<Mammal> clss( "Mammal" );
// some examples of what I've tried:
map[ "Animal::Mammal" ] = clss;
map[ "Animal::Mammal" ] = *clss;
map[ "Animal::Mammal" ] = &clss;
map[ "Animal::Mammal" ] = std::make_shared<Php::Class>( clss );
map[ "Animal::Mammal" ] = std::make_shared<Php::ClassBase>( clss );
map.insert( std::pair<std::string, Php::Class>( "Animal::Mammal", clss ) );
// etc., etc., etc. ...
但没有任何编译。
我遇到了从“operator*
未定义”到“不完整的类”(或类似的东西)到“模板参数无效”的编译错误。
正如你所看到的,我显然缺乏对这里需要做什么的实际理解。
如何确定地图的正确值类型应该是什么,然后如何正确地将“类”(它甚至是正确的类?)添加到地图中?
PS:我相当有信心我已经正确地将所有必要的头文件/文件包含在适当的文件中。为简洁起见,我只是在示例中省略了这些内容。
解决方案
它看起来像Php::Class
一个模板,所以你不能把它当作mapped_type
a std::map
,它不是一个类型。
您将必须为您正在注册的每个课程提供一个“注册我”功能,该功能将进入一个集合。
void registerMammal(Php::Namespace & ns) override
{
Php::Class<Animal::Mammal> cls("Mammal");
cls.method<&Animal::Mammal::method>("method");
// ...
ns.add(std::move(cls));
}
std::vector<std::function<void(Php::Extension &, Php::Namespace &)>> AnimalRegistrations{ registerMammal, ... };
PHPCPP_EXPORT void *get_module() {
static Php::Extension extension( "animals", "0.1" );
Php::Namespace ns( "Animal" );
for (auto & reg : AnimalRegistrations)
{
reg(ns);
}
extension.add( std::move( ns ) );
return extension;
}
如果您有大量要注册的命名空间,您最终可能会得到一个std::map<std::string, std::vector<std::function<void(Php::Namespace &)>>>
与该命名空间注册函数集合匹配的名称。
你会像这样使用它
Php::Extension extension( "dabbler", "0.1" );
for (auto & pair : NamespaceMap)
{
Php::Namespace ns(pair.first);
for (auto & reg : pair.second)
{
reg(ns);
}
extension.add( std::move( ns ) );
}
推荐阅读
- python - TypeError: raw_input() 接受 1 到 2 个位置参数,但给出了 4 个
- python - 基于另一个字典的布尔值创建字典列表的最pythonic方法
- sqlite - 如果存在其他 2 个表中的匹配项,则获取所有表值
- sitecore9 - Sitecore: override ChnagePassword class
- azure - Azure 数据工厂,如何从 scala(jar 作业)获取输出?
- r - R中对象的类和类型不同。我应该如何使其保持一致?
- amazon-emr - ModuleNotFoundError:emr 集群上没有名为“pyspark”的模块
- javascript - 需要在 Slack App Node.js 的异步函数中获取 REST API
- java - Intellij:可以在文本中搜索具有特定模式的文件
- powerbi - RLS 在数据集上成功测试,但在报告中不起作用