Which is more efficient: Return a value vs. Pass by reference?

后端 未结 7 1760
自闭症患者
自闭症患者 2020-11-28 08:37

I am currently studying how to write efficient C++ code, and on the matter of function calls, a question comes to mind. Comparing this pseudocode function:

n         


        
相关标签:
7条回答
  • 2020-11-28 08:48

    Returning the object should be used in most cases because of an optimsation called copy elision.

    However, depending on how your function is intended to be used, it may be better to pass the object by reference.

    Look at std::getline for instance, which takes a std::string by reference. This function is intended to be used as a loop condition and keeps filling a std::string until EOF is reached. Using the same std::string allows the storage space of the std::string to be reused in every loop iteration, drastically reducing the number of memory allocations that need to be performed.

    0 讨论(0)
  • 2020-11-28 08:50

    Performance-wise, copies are generally more expensive, although the difference might be negligible for small objects. Also, your compiler might optimize a return copy into a move, making equivalent to passing a reference.

    I'd recommend not passing non-const references unless you have a good reason to. Use the return value (e.g. functions of the tryGet() sort).

    If you want you can measure yourself the difference, as others have said already. Run the test code a few million times for both versions and see the difference.

    0 讨论(0)
  • 2020-11-28 09:02

    This pseudocode function:

    not-void function-name () {
        do-something
        return value;
    }
    

    would be better used when the returned value does not require any further modifications onto it. The parameter passed is only modified in the function-name. There are no more references required to it.


    otherwise-identical pseudocode function:

    void function-name (not-void& arg) {
        do-something
        arg = value;
    }
    

    would be useful if we have another method moderating the value of the same variable like and we need to keep the changes made to the variable by either of the call.

    void another-function-name (not-void& arg) {
        do-something
        arg = value;
    }
    
    0 讨论(0)
  • 2020-11-28 09:08

    First of all, take in account that returning an object will always be more readable (and very similar in performance) than having it passed by reference, so could be more interesting for your project to return the object and increase readability without having important performance differences. If you want to know how to have the lowest cost, the thing is what do you need to return:

    1. If you need to return a simple or basic object, the performance would be similar in both cases.

    2. If the object is so large and complex, returning it would need a copy, and it could be slower than having it as a referenced parameter, but it would spend less memory I think.

    You have to think anyway that compilers do a lot of optimizations which make both performances very similar. See Copy Elision.

    0 讨论(0)
  • 2020-11-28 09:10

    Some of the answers have touched on this, but I would like to emphasize in light of the edit

    For context, this question is limited to hardware platform-independent differences, and for the most part software too. Are there any machine-independent performance difference?

    If this is the limits of the question, the answer is that there is no answer. The c++ spec does not stipulate how either the return of an object or a passing by reference is implemented performance wise, only the semantics of what they both do in terms of code.

    A compiler is therefore free to optimize one to identical code as the other assuming this does not create a perceptible difference to the programmer.

    In light of this, I think it is best to use whichever is the most intuitive for the situation. If the function is indeed "returning" an object as the result of some task or query, return it, while if the function is performing an operation on some object owned by the outside code, pass by reference.

    You cannot generalize performance on this. As a start, do whatever is intuitive and see how well your target system and compiler optimizes it. If after profiling you will discover a problem, change it if you need to.

    0 讨论(0)
  • 2020-11-28 09:11

    Well, one must understand that compilation is not an easy buisness. there are many consideration taken when the compiler compiles your code.

    One can't simply answer this question because the C++ standard doesn't provide standard ABI (abstract binary interface), so each compiler is allowed to compile the code whatever it likes and you can get different results in each compilation.

    For example, on some projects C++ is compiled to managed extension of Microsoft CLR (C++/CX). since everything there is already a reference to an object on the heap, I guess there is not difference.

    The answer is not simpler for un-managed compilations. several quaestion come to mind when I think about "Will XXX run faster then YYY?", for example:

    • Is you object deafult-constructible?
    • Does your compiler support return-value-optimization?
    • Does your object support Copy-only semantics or both copy and move?
    • Is the object packed in contigious manner (e.g. std::array) or it has pointer to something on the heap? (e.g. std::vector)?

    If I give concrete example, my guess is that on MSVC++ and GCC, returning std::vector by value will be the as passing it by reference, because of r-value-optimization, and will be a bit (by few nanoseconds) faster then returning the vector by move. this may be completly different on Clang, for example.

    eventually, profiling is the only true answer here.

    0 讨论(0)
提交回复
热议问题