Safe to pass dereferenced anonymous shared_ptr?
data:image/s3,"s3://crabby-images/45e1e/45e1e700aa78ae42cb1c2026182d4faeba0c4ecd" alt=""
Q1: Is the below code dangerous?
A1: Only if the anonymous shared_ptr is deleted before function f returns.
Q2: Is the anonymous shared_ptr deleted before function f returns?
A2: ?
Thank you,
Chris
===
#include <iostream>
#include
data:image/s3,"s3://crabby-images/2f3a7/2f3a71cbdf809f126bec5afac8abbdf7ff830e30" alt=""
2013/11/6 Chris Stankevitz
Q1: Is the below code dangerous? A1: Only if the anonymous shared_ptr is deleted before function f returns.
Q2: Is the anonymous shared_ptr deleted before function f returns? A2: ?
Thank you,
Chris
===
#include <iostream> #include
void f(const int& i) { std::cout << "i = " << i << std::endl; }
int main() { f(*boost::make_shared<int>(42));
return 0; }
I don't think this particular use is dangerous, but does it make sense? I can't imagine why would you ever want to do that. In your (simplified I presume) example you don't share anything, you don't even need a pointer at all. In your real example you probably have some class with a user-defined constructor instead of int, but can't you do: f( int(42) ); HTH, Kris
data:image/s3,"s3://crabby-images/45e1e/45e1e700aa78ae42cb1c2026182d4faeba0c4ecd" alt=""
On Tue, Nov 5, 2013 at 3:30 PM, Krzysztof Czainski <1czajnik@gmail.com> wrote:
In your (simplified I presume) example you don't share anything, you don't even need a pointer at all. In your real example you probably have some class with a user-defined constructor instead of int, but can't you do:
Kris, Scott, Edward, Thank you for your replies regarding the safe use of the anonymous shared_ptrs. === Kris: FYI my more complex use-case looks like this: class TCImage { public: // Default constructs a black image TCImage(unsigned Width, unsigned Height); }; boost::shared_ptr<const std::string> GetJpegBytes(const TCImage& Image); GetJpegBytes generates JPEG bytes for an image and returns them as a shared_ptr. The shared_ptr is helpful because these JPEG bytes are passed around throughout my system. My question arose when I had the following atypical special-case (simplified but you get the idea): void WriteJpegBytes(const std::string& JpegBytes); void WriteBlackImage(); === I considered three options: 1. Pass a dereferenced anonymous boost::shared_ptr void WriteBlackImage() { WriteJpegBytes(*TCJpegEncoder::GetBytes(TCImage(640, 480))); } 2. Pass a dereferenced named boost::shared_ptr void WriteBlackImage() { boost::shared_ptr<const std::string> pBytes = TCJpegEncoder::GetBytes(TCImage(640, 480)); WriteJpegBytes(*pBytes); } 3. Change my interface: boost::shared_ptr<const std::string> GetJpegBytesAsSharedPointer(const TCImage& Image); std::string GetJpegBytesAsString(const TCImage& Image); I opted for approach (1) but wanted it clear it first with you guys. ===== Thank you again, Chris
data:image/s3,"s3://crabby-images/98c8b/98c8bcdf10354332a47d135fcce56d1b24c4fd71" alt=""
I have just one comment. I would recommend returning a unique_ptr instead of a shared_ptr, as you can then use the object as a unique_ptr, be move it into a shared_ptr, or adopt it by other means elsewhere in your data structures. Returning a shared_ptr locks you into keeping the data as a shared_ptr. Best Regards, M. Scott Mueller
On Nov 5, 2013, at 7:56 PM, "Chris Stankevitz"
wrote: Kris, Scott, Edward,
Thank you for your replies regarding the safe use of the anonymous shared_ptrs.
===
Kris: FYI my more complex use-case looks like this:
class TCImage { public:
// Default constructs a black image TCImage(unsigned Width, unsigned Height); };
boost::shared_ptr<const std::string> GetJpegBytes(const TCImage& Image);
GetJpegBytes generates JPEG bytes for an image and returns them as a shared_ptr. The shared_ptr is helpful because these JPEG bytes are passed around throughout my system.
data:image/s3,"s3://crabby-images/2f3a7/2f3a71cbdf809f126bec5afac8abbdf7ff830e30" alt=""
2013/11/6 Scott Mueller
I have just one comment. I would recommend returning a unique_ptr instead of a shared_ptr, as you can then use the object as a unique_ptr, be move it into a shared_ptr, or adopt it by other means elsewhere in your data structures. Returning a shared_ptr locks you into keeping the data as a shared_ptr.
Best Regards,
M. Scott Mueller
If you're in C++11, why would you use a pointer at all and not return a string? It's movable. If you're not in C++11, then consider returning by value anyway. RVO will most likely avoid the copy anyway. That said, (I wish unique_ptr emulation was in boost) I get it why use shared_ptr even though you don't share anything... Regards, Kris
data:image/s3,"s3://crabby-images/2f3a7/2f3a71cbdf809f126bec5afac8abbdf7ff830e30" alt=""
2013/11/6 Krzysztof Czainski <1czajnik@gmail.com>
If you're not in C++11, then consider returning by value anyway. RVO will most likely avoid the copy anyway. That said, (I wish unique_ptr emulation was in boost) I get it why use shared_ptr even though you don't share anything...
Or maybe return a boost::container::vector<char> in C++98? It's movable, you know ;-) HTH, Kris
data:image/s3,"s3://crabby-images/98c8b/98c8bcdf10354332a47d135fcce56d1b24c4fd71" alt=""
Hello,
Q1: Is the below code dangerous?
A1: Short answer: no. For the long answer, see the end of this email.
Q2: Is the anonymous shared_ptr deleted before function f returns?
A2: No, it is deleted afterward.
-----Original Message-----
From: Boost-users [mailto:boost-users-bounces@lists.boost.org] On Behalf Of Chris Stankevitz
Sent: Tuesday, November 05, 2013 3:13 PM
To: boost-users@lists.boost.org
Subject: [Boost-users] Safe to pass dereferenced anonymous shared_ptr?
...
#include <iostream>
#include
From 12.2.3 of ISO/IEC 14882:2011(E): When an implementation introduces a temporary object of a class that has a non-trivial constructor (12.1, 12.8), it shall ensure that a constructor is called for the temporary object. Similarly, the destructor shall be called for a temporary with a non-trivial destructor (12.4). Temporary objects are destroyed as the last step in evaluating the full-expression (1.9) that (lexically) contains the point where they were created. This is true even if that evaluation ends in throwing an exception. The value computations and side effects of destroying a temporary object are associated only with the full-expression, not with any specific subexpression.
Yep, I paid the 30 bucks. ;) Best regards, M. Scott Mueller
data:image/s3,"s3://crabby-images/6d56c/6d56cbeeeb9fb0c666908dd23c3154bc129dd5c6" alt=""
On 11/5/2013 6:12 PM, Chris Stankevitz wrote:
Q1: Is the below code dangerous? A1: Only if the anonymous shared_ptr is deleted before function f returns.
Q2: Is the anonymous shared_ptr deleted before function f returns? A2: ?
Thank you,
Chris
===
#include <iostream> #include
void f(const int& i) { std::cout << "i = " << i << std::endl; }
int main() { f(*boost::make_shared<int>(42));
return 0; }
A2: No. But this is C++ and has nothing to do with shared_ptr.
participants (4)
-
Chris Stankevitz
-
Edward Diener
-
Krzysztof Czainski
-
Scott Mueller