-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathSharedPtr_SmartPointer2.cpp
105 lines (89 loc) · 2.56 KB
/
SharedPtr_SmartPointer2.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
/**
* More tests for std::shared_ptr.
* Inspired by https://docs.microsoft.com/en-us/cpp/cpp/how-to-create-and-use-shared-ptr-instances?view=msvc-160
*/
#include <iostream>
#include <vector>
#include <memory>
#include <cassert>
struct WidgetBase
{
virtual ~WidgetBase() = default;
};
struct Widget1 : public WidgetBase
{
Widget1(int val, const char* name):
WidgetBase(),
m_val(val),
m_name(name)
{}
virtual ~Widget1() { std::cout << "dtor(Widget1): " << m_val << ", " << m_name << "\n"; }
int m_val;
const char* m_name;
};
struct Widget2 : public WidgetBase
{
Widget2(int size, float val):
WidgetBase(),
m_size(size),
m_val(val)
{}
virtual ~Widget2() { std::cout << "dtor(Widget2): " << m_size << ", " << m_val << "\n"; }
int m_size;
float m_val;
};
// argument is by copy
static void PrintWidget1(const Widget1 copy)
{
std::cout << "m_val: " << copy.m_val << ", m_name: " << copy.m_name << "\n";
// after this, a copy instance will be destroyed and thus immediately decrement use_count back to 1, see console output of "dtor(Widget1)..."
}
// argument is by pointer
static void PrintWidget1(const Widget1* ptr)
{
std::cout << "m_val: " << ptr->m_val << ", m_name: " << ptr->m_name << "\n";
}
// argument is by reference
static void PrintWidget2(const Widget2& w)
{
std::cout << "m_size: " << w.m_size << ", m_val: " << w.m_val << "\n";
}
int main()
{
std::vector<std::shared_ptr<WidgetBase>> widgets {
std::make_shared<WidgetBase>(),
std::make_shared<Widget1>(1, "Widget1_1"),
std::make_shared<Widget1>(2, "Widget1_2"),
std::make_shared<Widget2>(10, 10.5),
std::make_shared<Widget2>(20, 20.5)
};
// before end of life, number of shared_ptr owns it (use_count()) should just be 1 for each
for (std::size_t i=0; i<widgets.size(); ++i)
{
assert(widgets[i].use_count() == 1);
}
// test local scope lifetime
{
std::shared_ptr<WidgetBase> wAnother = widgets[4];
assert(widgets[4].use_count() == 2);
assert(wAnother.use_count() == 2);
}
assert(widgets[4].use_count() == 1);
// test range base via copy
// ref count is incremented for each
for (const auto w : widgets)
{
assert(w.use_count() == 2);
}
// test range base via reference
// ref count is **not** incremented
for (const auto& w : widgets)
{
assert(w.use_count() == 1);
}
// test getting and sending underlying pointer of std::shared_ptr to function
PrintWidget1(*std::static_pointer_cast<const Widget1>(widgets[1]));
PrintWidget1(std::static_pointer_cast<const Widget1>(widgets[1]).get());
PrintWidget2(*std::static_pointer_cast<const Widget2>(widgets[3]));
return 0;
}