Does somebody have any idead on how to pass boost::shared_ptr
- by value or by reference.
On my platform (32bit) sizeof(shared_ptr)
equals 8 bytes and it looks like I should pass them by reference, but maybe somebody has another opinion / did a profile / something like that?
You can see this in two ways:
a
boost::shared_ptr
is an object (and should be passed by const &).a
boost::shared_ptr
models a pointer and should be treated as a pointer.Both of them are valid, and the second option will incur the cost of constructing and destructing the additional instance (which shouldn't be a big issue unless you're writing performance-critical code).
There are two cases when you should pass by value:
when you're not sure that the type of the pointer will remain the same. That is, if you are considering replacing the
shared_ptr<T>
withT*
in your codebase somewhere in the future, it makes sense to writetypedef shared_ptr<T> TPtr;
and define your functions asvoid yourfunction(TPtr value)
In this case, when you change the pointer type you will only have to modify the
typedef
line.when you are sharing a pointer between two modules with a different lifetime. In that case, you need to make sure you have two instances of the smart pointer, and that both increment the reference count of the pointer.
In other cases, it's a matter of preference (unless you're writing performance-critical code in which case different rules apply).
I believe the essence of shared_ptr is that it is an object, and should be passed around by value. As I recall, it automatically handles reference counting using the ctor & dtor, so it knows when to free the pointer.
Whenever I've used it, I passed it by value.
In C++, it's generally a bad idea to choose whether to pass by value or reference based on the object's size.
Partly because the compiler will often perform copy elision on pass-by-value, negating the cost of copying the value, but mainly because the two options often behave differently.
So choose the option which bests expresses what you need to do.
With a
shared_ptr
, its entire reason for existing is that it can be copied, so that multiple objects can share ownership of the pointed-to object. If you never pass ashared_ptr
by value, you can start wondering why it is ashared_ptr
at all. Ascoped_ptr
may be a more efficient solution then.Obviously, that's not to say you should always pass
shared_ptr
's by value either. Just that pass-by-value is a common use case for them.If you need the caller and callee to have shared ownership, pass by value. If you don't want the callee to take any kind of ownership, pass by reference.
I would pass a
shared_ptr
by reference-to-const. If you pass it by value, the use count is incremented by entering the function and decremented by leaving it, which is unnecessary overhead.I'd call reference on a shared_ptr, where you can. This is because shared_ptr's operations are atomic, and thus unusually high overhead, as well as the potential to cache miss on the increment/decrement that simply isn't necessary if the calling function still has a strong reference to the object.
Of course, this isn't always possible- for example, within containers or in member vars, which need to hold their own reference.
And as another answerer said, the vast majority of any overhead in C++ is in the con/destructor, not passing the raw bits and bytes. This is true of shared_ptr, too.