C ++ 11 — Как симметрично реализовать сериализацию и десериализацию шаблонных функций в переполнении стека

Я хочу написать серию шаблонных функций для сериализации и десериализации объектов. Я закончил сериализацию и все работает:

#ifndef SERIALIZE_H
#define SERIALIZE_H

#include <string>
#include <vector>
#include <unordered_set>
#include <unordered_map>
#include <memory>

inline std::string to_json(int value) {
return std::to_string(value);
}inline std::string to_json(long value) {
return std::to_string(value);
}inline std::string to_json(double value) {
return std::to_string(value);
}inline std::string to_json(const std::string& myStr) {
return "\"" + myStr + "\"";
}template <typename T>
std::string to_json(const std::vector<T>& vec) {
std::string json("[");

for(auto &i : vec) {
json += to_json(i);
json += ",";
}

if (!vec.empty()) json.pop_back();
json += "]";
return json;
}template <typename T>
std::string to_json(const std::unordered_set<T>& mySet) {
std::string json("[");

for(const auto& i : mySet) {
json += to_json(i);
json += ",";
}

if (!mySet.empty()) json.pop_back();
json += "]";
return json;
}template <typename K, typename V>
std::string to_json(const std::unordered_map<K, V>& myMap) {
std::string json("{");

for(const auto& i : myMap) {
json += to_json(i.first);
json += ":";
json += to_json(i.second);
json += ",";
}

if (!myMap.empty()) json.pop_back();
json += "}";
return json;
}

#endif //SERIALIZE_H

это serialize.h можно сериализовать все виды комбинаций, такие как unordered_map<string, vector<int>>,

Теперь я не знаю, как рекурсивно реализовать функции десериализации для поддержки произвольных комбинаций.

Следующее мое deserialize.h который не работает:

#ifndef DESERIALIZE_H
#define DESERIALIZE_H

#include <string>
#include <rapidjson/document.h>template<typename T>
T from_json(const std::string &json);

template<>
int from_json(const std::string &json) {
return std::stoi(json);
}

template<>
long from_json(const std::string &json) {
return std::stol(json);
}

template<>
double from_json(const std::string &json) {
return std::stod(json);
}

template<>
std::string from_json(const std::string &json) {
return json.substr(1, json.size()-1);
}

//
template<typename T>
std::vector<T> from_json(const std::string& json) {
rapidjson::Value jsonValue;
{
const std::string &input = "{\"input\":" + json + "}";
rapidjson::Document document;
document.Parse(input.c_str());
jsonValue = document["input"];
};
std::vector<T> vec;
assert(jsonValue.IsArray());

for (rapidjson::SizeType i = 0; i < jsonValue.Size(); i++) {
int element = from_json<T>(std::string(jsonValue[i].GetString()));
vec.push_back(element);
}
return vec;
}

#endif //DESERIALIZE_H

rapidjson библиотека JSON C ++, https://github.com/miloyip/rapidjson

Затем, если я попытаюсь десериализовать строку JSON:

#include "deserialize.h>

int main() {
auto vec1 = from_json<std::vector<int>>(std::string("[1,2,3]"));
return 0;
}

Это выбросит ошибки компиляции:

error: call of overloaded ‘from_json(std::string)’ is ambiguous

Кажется, нет способа реализовать функцию десериализации так же легко, как сериализация.

Есть идеи?

0

Решение

Вы пытаетесь перегрузить одну и ту же функцию (from_json (…)) с одинаковыми аргументами и разным типом возврата каждый раз. Это не законно.
Для перегрузки вам нужны разные типы аргументов или номер аргумента.

Instead of
template<>
double from_json(const std::string &json) {
return std::stod(json);
}

template<>
std::string from_json(const std::string &json) {
return json.substr(1, json.size()-1);
}

возможно попробуйте это, или по крайней мере это должно быть общей идеей:

template<>
void from_json(const std::string &json, double *out_value) {
*out_value = std::stod(json);
}

template<>
void from_json(const std::string &json, std::string *out_value) {
out_value = json.substr(1, json.size()-1);
}

Я уверен, что есть ошибки, но я думаю, что это может работать (если вы их исправите)

2

Другие решения


По вопросам рекламы [email protected]