-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathExceptionsSample.cpp
174 lines (146 loc) · 7.14 KB
/
ExceptionsSample.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
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
#include <iostream>
#include <ostream>
#include <memory>
#include <exception>
#include <stdexcept>
#include <typeinfo>
#include <algorithm>
#include <cstdlib>
#include "InvalidArgumentException.h"
#include "../pchar.h"
using namespace CppForCsExceptions;
using namespace std;
class ThrowClass {
public:
ThrowClass(void) : m_shouldThrow(false) {
wcout << L"Constructing ThrowClass." << endl;
}
explicit ThrowClass(bool shouldThrow) : m_shouldThrow(shouldThrow) {
wcout << L"Constructing ThrowClass. shouldThrow = "
<< (shouldThrow ? L"true." : L"false.") << endl;
if (shouldThrow) {
throw InvalidArgumentException<const char*>(
"ThrowClass",
"ThrowClass(bool shouldThrow)",
"shouldThrow",
"true"
);
}
}
~ThrowClass(void) { wcout << L"Destroying ThrowClass." << endl; }
const wchar_t* GetShouldThrow(void) const {
return (m_shouldThrow ? L"True" : L"False");
}
private:
bool m_shouldThrow;
};
class RegularClass {
public:
RegularClass(void) { wcout << L"Constructing RegularClass." << endl; }
~RegularClass(void) { wcout << L"Destroying RegularClass." << endl; }
};
class ContainStuffClass {
public:
ContainStuffClass(void) : m_regularClass(new RegularClass()),
m_throwClass(new ThrowClass()) {
wcout << L"Constructing ContainStuffClass." << endl;
}
ContainStuffClass(const ContainStuffClass& other) :
m_regularClass(new RegularClass(*other.m_regularClass)),
m_throwClass(other.m_throwClass) {
wcout << L"Copy constructing ContainStuffClass." << endl;
}
~ContainStuffClass(void) {
wcout << L"Destroying ContainStuffClass." << endl;
}
const wchar_t* GetString(void) const { return L"I'm a ContainStuffClass."; }
private:
unique_ptr<RegularClass> m_regularClass;
shared_ptr<ThrowClass> m_throwClass;
};
void TerminateHandler(void) {
wcout << L"Terminating due to unhandled exception." << endl;
// Если вызвать abort (из <cstdlib>), то программа завершится аномально.
// Если ничего не вызывать для того чтобы выйти из этого метода, то программа
// также завершится аномально.
abort();
// Можно вызвать exit(0) (также из <cstdlib>), тогда программа выйдет так
// как будто ничего плохого не произошло. Это плохая практика потому что
// кое-что плохое произошло. Но тем не менее это возможно, бросить
// необрабатываемое исключение и всё же выйти таким образом, чтобы это
// не интерпретировалось как сбой. Если программа внезапно завершается, без
// сбоя, то возможно в ней есть что-то подобное.
// exit(0);
}
int _pmain(int /*argc*/, _pchar* /*argv*/[]) {
// Установка кастомного обработчика для std::terminate. Этот обработчик
// сработает, если запустить программу из командной строки. Отладчик
// Visual Studio перехватит необработанное исключение и предоставит средства
// для отладки.
set_terminate(&TerminateHandler);
try {
ContainStuffClass cSC;
// ВЫВОД:
// Constructing RegularClass.
// Constructing ThrowClass.
// Constructing ContainStuffClass.
wcout << cSC.GetString() << endl;
// ВЫВОД:
// I'm a ContainStuffClass.
ThrowClass tC(false);
// ВЫВОД:
// Constructing ThrowClass. shouldThrow = false.
wcout << L"tC should throw? " << tC.GetShouldThrow() << endl;
// ВЫВОД:
// tC should throw? False
tC = ThrowClass(true);
// ВЫВОД:
// Constructing ThrowClass. shouldThrow = true.
// Destroying ThrowClass.
// Destroying ContainStuffClass.
// Destroying ThrowClass.
// Destroying RegularClass.
wcout << L"tC should throw? " << tC.GetShouldThrow() << endl;
}
// Одним из недостатков использования шаблонов для исключений является то,
// что нужнен catch-обработчик для каждой возможной специализации. Если
// только нет базового класса от которого они все унаследованы. И он в
// данном случае есть. Чтобы избежать отлова других std::invalid_argument
// исключений, мы создали абстрактный класс InvalidArgumentExceptionBase
// который единственно служит как базовый класс для всех специализаций
// InvalidArgumentException<T>. Теперь мы можем отлавливать их все
// (если нужно) без необходимости писать catch-обработчик для каждого по
// отдельности. Однако при необходимости можно написать обработчик для
// каждой отдельной специализации.
catch (InvalidArgumentExceptionBase& e) {
wcout << L"Caught '" << typeid(e).name() << L"'." << endl
<< L"Message: " << e.what() << endl;
// ВЫВОД:
// Caught 'N18CppForCsExceptions24InvalidArgumentExceptionIPKcEE'.
// Message: ThrowClass::ThrowClass(bool shouldThrow) -
// parameter 'shouldThrow' had invalid value 'true'.
}
// Отлов исключений унаследованных от std::exception, для которых у нас нет
// специализированного обработчика. Так как непонятно, что это, то нужно
// поймать это, занести в журнал и перебросить.
catch (std::exception& e) {
wcout << L"Caught '" << typeid(e).name() << L"'." << endl
<< L"Message: " << e.what() << endl;
// Простая инструкция trow означает переброс.
throw;
}
// Этот catch отлавливает всё, независимо от типа. Также как и с отловом
// System.Exception в C#, этот отлов нужен только для того чтобы перебросить
// исключение.
catch (...) {
wcout << L"Caught unknown exception type." << endl;
throw;
}
// Эта строка приветет к запуску кастомного обработчика терминации:
wcout << L"tC should throw? " << ThrowClass(true).GetShouldThrow() << endl;
// ВЫВОД:
// tC should throw? Constructing ThrowClass. shouldThrow = true.
// Terminating due to unhandled exception.
// Аварийный останов (стек памяти сброшен на диск)
return 0;
}