如何正确地完善前进"吸气功能?

时间:2014-09-29 10:00:17

标签: c++ c++11 move c++14 perfect-forwarding

我正在为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。

2 个答案:

答案 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)