我正在尝试使用SFINAE编写模板类,以从地图转换为json,反之亦然。想法是将类限制为具有整数值类型或可转换为int(即枚举)的映射,但是我在编译类时遇到问题。
到目前为止,这是我的代码:
MapToJsonConvertor.h:
#ifndef COMMON_MAPTOJSONCONVERTOR_H_
#define COMMON_MAPTOJSONCONVERTOR_H_
#include <Poco/JSON/Object.h>
#include <Poco/JSON/Parser.h>
#include <Poco/Dynamic/Var.h>
#include <Poco/JSON/Stringifier.h>
#include <map>
#include <type_traits>
template<typename T,
typename = typename std::enable_if<std::is_convertible<T, int>::value, T>::type>
class MapToJsonConvertor
{
public:
MapToJsonConvertor(const std::map<int, T> &mapFrom): mMap(mapFrom)
{}
MapToJsonConvertor(const Poco::JSON::Object::Ptr &jsonObject): mJsonObject(jsonObject)
{}
Poco::JSON::Object::Ptr Convert(const std::string &rootName);
//std::map<int, T> Convert(const Poco::JSON::Object::Ptr &jsonObject);
private:
std::map<int, T> mMap;
Poco::JSON::Object::Ptr mJsonObject;
};
#endif /* COMMON_MAPTOJSONCONVERTER_H_ */
MapToJsonConvertor.cpp
#include "MapToJsonConvertor.h"
template<typename T>
Poco::JSON::Object::Ptr MapToJsonConvertor<T>::Convert(const std::string &rootName)
{
Poco::JSON::Object::Ptr root = new Poco::JSON::Object();
Poco::JSON::Array::Ptr array = new Poco::JSON::Array();
if (!mMap.empty())
{
for (auto &elem : mMap)
{
Poco::JSON::Object::Ptr elemPtr = new Poco::JSON::Object();
elemPtr->set("key", elem.first);
elemPtr->set("value", static_cast<int>(elem.second));
array->add(elemPtr);
}
}
root->set("elements", array);
return root;
}
我得到的错误是:
../src/Common/MapToJsonConvertor.cpp:12:83:错误:无效使用
类型不完整的“类MapToJsonConvertor ” Poco :: JSON :: Object :: Ptr
MapToJsonConvertor :: Convert(const std :: string&rootName)
^
在../src/Common/MapToJsonConvertor.cpp:9:0中包含的文件中:
../src/Common/MapToJsonConvertor.h:21:7:注意:“ class”的声明
MapToJsonConvertor ’类MapToJsonConvertor
^
我正在使用Poco库进行Json构建。
我仍然对使用SFINAE进行模板元编程还不陌生,因此希望获得一些帮助。我究竟做错了什么?
最佳答案
因此,首先,应该按照here所述在头文件中实现模板。
除此之外,您的问题还在于源文件中函数定义的模板签名。您正在类中为SFINAE使用第二个模板参数,因此您还必须在定义中指定该模板参数:
template<typename T, typename U> // note the 2nd parameter
Poco::JSON::Object::Ptr MapToJsonConvertor<T, U>::Convert(const std::string &rootName)
{
...
}