内容简介:So in ye olde days, before C++11 and move semantics, it was common for functions to use mutable references to pass container-content to the caller, like this:and you would often use it like this:Basically trading expressiveness and convenience for speed/ef
So in ye olde days, before C++11 and move semantics, it was common for functions to use mutable references to pass container-content to the caller, like this:
void random_between(std::vector<int>& out, int left, int right, std::size_t N) { std::uniform_int_distribution<> distribution(left, right); for (std::size_t i = 0; i < N; ++i) out.push_back(distribution(rng)); }
and you would often use it like this:
std::vector<int> numbers; random_between(numbers, 7, 42, 10);
Basically trading expressiveness and convenience for speed/efficiency.
Convenience is king
Now obviously, those days are over. With move-semantics and guaranteed copy-elision backing us up, it is usually fine to just return the filled container, like this:
std::vector<int> random_between(int left, int right, std::size_t N) { std::vector<int> out; std::uniform_int_distribution<> distribution(left, right); for (std::size_t i = 0; i < N; ++i) out.push_back(distribution(rng)); return out; }
Now you no longer have to initialize the container to use this function and the function also became pure, clearly differentiating between its inputs and outputs.
Mostly better?
However, there is a downside: Before, the function could be used to append multiple runs into the same container, like this:
std::vector<int> numbers; for (int i = 0; i < 5; ++i) random_between(numbers, 50*i + 7, 50*i + 42, 10);
That use case suddenly became a lot harder. Also, what if you want to keep your vector around and just .clear()
it before calling the function again later, to save allocations? That’s also no longer possible. I am not saying that these two use cases should make you prefer the old variant, as they tend not to happen very often. But when they do, it’s all the more annoying. So what if we could have your cake and eat it, too?
A Compromise
How about this:
std::vector<int> random_between(int left, int right, std::size_t N, std::vector<int> out = {}) { std::uniform_int_distribution<> distribution(left, right); for (std::size_t i = 0; i < N; ++i) out.push_back(distribution(rng)); return out; }
Now you can use it to just append again:
std::vector<int> numbers; for (int i = 0; i < 5; ++i) numbers = random_between( 50*i + 7, 50*i + 42, 10, std::move(numbers));
But you can also use it in the straightforward way, for the hopefully more common case:
auto numbers = random_between( 50*i + 7, 50*i + 42, 10);
Now you should definitely not do this with all your functions returning a container. But it is a nice pattern to have up your sleeve when the need arises. It should be noted that passing a mutable reference can still be faster in some cases, as that will save you two moves. And you can also add a container-returning facade variant as an overload, but I think this pattern is a very nice compromise that can be implemented by moving a single variable to the parameter list and defaulting it. It keeps 99% of the use cases identically to the original container-returning variant, while making the “append” use slightly more verbose, but also more expressive.
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
劫持
玛丽•K. 斯温格尔(Mari K. Swingle) / 邓思渊 / 中信出版集团股份有限公司 / 2018-5-1 / CNY 59.00
《劫持》是一本探讨人与科技的关系的书,基于一位心理学博士20年的临床经验及其作为神经认知科学研究者的脑—电研究成果。在这本面向大众的科普书中,作者以深入浅出的方式,探讨了手机、电脑等便携式数字设备及让人“永不下线”的互联网对现代人尤其是青少年大脑的影响,从神经认知科学和精神分析的角度,有力地证明了数字媒介与大脑和人类行为的关系,探讨了手机等如何对人的大脑进行劫持或操控,并给出了自己作为从业医师的实......一起来看看 《劫持》 这本书的介绍吧!