我正在为C ++ 14创建一个JSON库,我试图尽可能地使用移动语义。
我的Value
课程有几个主持人,他们总是尽可能地尝试移动:
template<class T> void setObj(T&& x) { type = Obj; hObj.init(forward<T>(x)); }
template<class T> void setArr(T&& x) { type = Arr; hArr.init(forward<T>(x)); }
template<class T> void setStr(T&& x) { type = Str; hStr.init(forward<T>(x)); }
auto& getObj() & noexcept { assert(is<Obj>()); return hObj; }
auto& getArr() & noexcept { assert(is<Arr>()); return hArr; }
auto& getStr() & noexcept { assert(is<Str>()); return hStr; }
const auto& getObj() const& noexcept { assert(is<Obj>()); return hObj; }
const auto& getArr() const& noexcept { assert(is<Arr>()); return hArr; }
const auto& getStr() const& noexcept { assert(is<Str>()); return hStr; }
auto getObj() && noexcept { assert(is<Obj>()); return move(hObj); }
auto getArr() && noexcept { assert(is<Arr>()); return move(hArr); }
auto getStr() && noexcept { assert(is<Str>()); return move(hStr); }
从代码中可以看出,使用模板和通用引用可以很容易地完善转发setter函数。
如何对getter函数执行相同的操作?我很确定我必须使用模板返回类型,但我不确定如何复制ref-qualifiers和const-correctness。
答案 0 :(得分:4)
由于你无法对ref限定符和成员常量进行模板化,所以可悲的答案是你无法做到。你必须写出来。
答案 1 :(得分:2)
这不像C ++模板,但它可以完成任务。
#define GETTERS(V) \
V(Obj) \
V(Arr) \
V(Str)
#define VISIT(X) \
auto &get ## X() & noexcept { assert(is<X>()); return h ## Obj; } \
const auto &get ## X() const& noexcept { assert(is<X>()); return h ## Obj; } \
auto &get ## X() && noexcept { assert(is<X>()); return std::move(h ## Obj); }
GETTERS(VISIT)