boost exception provides a new exception type, that lets you add data to an exception after it has been thrown.ios
#include <boost/exception/all.hpp> #include <exception> #include <new> #include <string> #include <algorithm> #include <limits> #include <iostream> typedef boost::error_info<struct tag_errmsg, std::string> errmsg_info; struct allocation_failed : public boost::exception, public std::exception { const char *what() const noexcept { return "allocation failed"; } }; char *allocate_memory(std::size_t size) { char *c = new (std::nothrow) char[size]; if (!c) throw allocation_failed{}; return c; } char *write_lots_of_zeros() { try { char *c = allocate_memory(std::numeric_limits<std::size_t>::max()); std::fill_n(c, std::numeric_limits<std::size_t>::max(), 0); return c; } catch (boost::exception &e) { e << errmsg_info{"writing lots of zeros failed"}; throw; } } int main() { try { char *c = write_lots_of_zeros(); delete[] c; } catch (boost::exception &e) { std::cerr << boost::diagnostic_information(e); } return 0; }
output:ide
Throw location unknown (consider using BOOST_THROW_EXCEPTION) Dynamic exception type: struct allocation_failed std::exception::what: allocation failed [struct tag_errmsg *] = writing lots of zeros failed
unction write_lots_of_zeros() calls allocate_memory(). allocate_memory() allocates memory dynamically. The function passes std::nothrow to new and checks whether the return values is 0. if memory allocation fails, an exception of type allocation_failed is thrown.ui
write_lots_of_zeros()
calls allocate_memory()
to try and allocate a memory block with the greatest possible size. This is done with the help of max()
from std::numeric_limits
. The example intentionally tries to allocate that much memory to make the allocation fail.spa
With Boost.Exception, data can be added to an exception at any time. You just need to define a type based on boost::error_info
for each bit of data you need to add.code
boost::error_info is a template that expects two parameters. The first parameter is a atag that uniquely identifies the newly created type. This is typically a structure with a unique name. The second parameter refers to the type of the value stored inside the exception.orm
In the catch
handler of write_lots_of_zeros()
, errmsg_info
is used to create an object that is initialized with the string 「writing lots of zeros failed」. This object is then added to the exception of type boost::exception
using operator<<
. Then the exception is re-thrown.blog
2. BOOST_THROW_EXCEPTIONci
char *allocate_memory(std::size_t size) { char *c = new (std::nothrow) char[size]; if (!c) BOOST_THROW_EXCEPTION(allocation_failed{}); return c; }
output:string
main.cpp(20): Throw in function char *__cdecl allocate_memory(unsigned int) Dynamic exception type: class boost::exception_detail::clone_impl<struct boost::exception_detail::error_info_injector<struct allocation_failed> > std::exception::what: allocation failed [struct tag_errmsg *] = writing lots of zeros failed
using the macro BOOST_THROW_EXCEPTION instead of throw, data such as function name, file name, and line number are automatically added to the exception.it
BOOST_THROW_EXCEPTION accesses the function boost::enable_error_info(), which identifies whether or not an exception is derived from boost::exception. If not, it creates a new exception type derived from the specified type and boost::exception.