35

Is it possible to return a standard container from a function without making a copy?

Example code:

std::vector<A> MyFunc();

...

std::vector<A> b = MyFunc();

As far as I understand, this copies the return value into a new vector b. Does making the function return references or something like that allow avoiding the copy?

static_rtti
  • 53,760
  • 47
  • 136
  • 192

3 Answers3

32

If your compiler supports the NRVO then no copy will be made, provided certain conditions are met in the function returning the object. Thankfully, this was finally added in Visual C++ 2005 (v8.0) This can have a major +ve impact on perf if the container is large, obviously.

If your own compiler docs do not say whether or not it's supported, you should be able to compile the C++ code to assembler (in optimized/release mode) and check what's done using a simple sample function.

There's also an excellent broader discussion here

Community
  • 1
  • 1
Steve Townsend
  • 53,498
  • 9
  • 91
  • 140
  • Thanks! Any idea about NRVO in gcc? – static_rtti Sep 15 '10 at 20:26
  • @static_rtti - I will have to defer to Linux folks for that one, for fear of putting foot in mouth – Steve Townsend Sep 15 '10 at 20:36
  • You can easily test with a class that traces in the Copy CTor. Note that this is a "can be" optimization, that won't necessarily work e.g. with differentrly named return objects. However, with C++0x rvalue references, classes can implement a guarantee. You can expect this for standard containers in an up-to-date STL. – peterchen Sep 15 '10 at 21:00
  • 2
    AFAIK, with GCC you'll have to turn this optimization off with -fno-elide-constructors, as it is otherwise enabled even with -O0. – UncleBens Sep 15 '10 at 21:11
  • 2
    @static_rtti: GCC is very good at eliding unnecessary copies. As far as I know, *no* other compiler is *better* at this -- just equally good or worse. – sellibitze Sep 15 '10 at 21:44
  • 1
    Agree with peterchen, there are a number of situations NRVO won't kick in, outlined in your linked document. I think the answer should discuss C++0x move semantics to deserve being the top-voted and chosen answer, so unfortunately I'm downvoting this. The compiler is never required to perform any optimization - with move semantics you can require the compiler not make a copy. – AshleysBrain Sep 15 '10 at 22:26
19

Rvalues ("temporaries") bound to const references will have their lifetime extended to the end of the reference's lifetime. So if you don't need to modify that vector, the following will do:

const std::vector<A>& b = MyFunc();

if you need to modify the vector, just code it the way that's easiest to read until your have proof (obtained through profiling) that this line even matters performance-wise.

Otherwise rely on C++1x with its rvalue references and move semantics coming along "real soon now" and optimizing that copy out without you having to do anything.

sbi
  • 219,715
  • 46
  • 258
  • 445
3

If you can modify the signature of the function then you can use

std::vector<A>& MyFunc(); 

or

void MyFunc(std::vector<A>& vect);

You could also return a smart pointer, but that involves newing the object.

some_smart_pointer<std::vector<A>> MyFunc();

HTH

Eclipse
  • 44,851
  • 20
  • 112
  • 171
beezler
  • 646
  • 6
  • 18