IM/include/mysql++/noexceptions.h
2023-05-03 19:09:14 +08:00

164 lines
5.4 KiB
C++

/// \file noexceptions.h
/// \brief Declares interface that allows exceptions to be optional
///
/// A class may inherit from OptionalExceptions, which will add to it
/// a mechanism by which a user can tell objects of that class to
/// suppress exceptions. (They are enabled by default.) This module
/// also declares a NoExceptions class, objects of which take a
/// reference to any class derived from OptionalExceptions. The
/// NoExceptions constructor calls the method that disables exceptions,
/// and the destructor reverts them to the previous state. One uses
/// the NoExceptions object within a scope to suppress exceptions in
/// that block, without having to worry about reverting the setting when
/// the block exits.
/***********************************************************************
Copyright (c) 2005-2007 by Educational Technology Resources, Inc.
Others may also hold copyrights on code in this file. See the
CREDITS.txt file in the top directory of the distribution for details.
This file is part of MySQL++.
MySQL++ is free software; you can redistribute it and/or modify it
under the terms of the GNU Lesser General Public License as published
by the Free Software Foundation; either version 2.1 of the License, or
(at your option) any later version.
MySQL++ is distributed in the hope that it will be useful, but WITHOUT
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
License for more details.
You should have received a copy of the GNU Lesser General Public
License along with MySQL++; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301
USA
***********************************************************************/
#ifndef MYSQLPP_NOEXCEPTIONS_H
#define MYSQLPP_NOEXCEPTIONS_H
#include "common.h"
namespace mysqlpp {
#if !defined(DOXYGEN_IGNORE)
// Make Doxygen ignore this
class MYSQLPP_EXPORT NoExceptions;
#endif
/// \brief Interface allowing a class to have optional exceptions.
///
/// A class derives from this one to acquire a standard interface for
/// disabling exceptions, possibly only temporarily. By default,
/// exceptions are enabled.
///
/// Note that all methods are const even though some of them change our
/// internal flag indicating whether exceptions should be thrown. This
/// is justifiable because this is just an interface class, and it
/// changes the behavior of our subclass literally only in exceptional
/// conditions. This Jesuitical interpretation of "const" is required
/// because you may want to disable exceptions on const subclass
/// instances.
///
/// If it makes you feel better about this, consider that the real
/// change isn't within the const OptionalExceptions subclass instance.
/// What changes is the code wrapping the method call on that instance
/// that can optionally throw an exception. This outside code is in
/// a better position to say what "const" means than the subclass
/// instance.
class MYSQLPP_EXPORT OptionalExceptions
{
public:
/// \brief Default constructor
///
/// \param e if true, exceptions are enabled (this is the default)
OptionalExceptions(bool e = true) :
exceptions_(e)
{
}
/// \brief Destroy object
virtual ~OptionalExceptions() { }
/// \brief Enable exceptions from the object
void enable_exceptions() const { exceptions_ = true; }
/// \brief Disable exceptions from the object
void disable_exceptions() const { exceptions_ = false; }
/// \brief Returns true if exceptions are enabled
bool throw_exceptions() const { return exceptions_; }
protected:
/// \brief Sets the exception state to a particular value
///
/// This method is protected because it is only intended for use by
/// subclasses' copy constructors and the like.
void set_exceptions(bool e) const { exceptions_ = e; }
/// \brief Declare NoExceptions to be our friend so it can access
/// our protected functions.
friend class NoExceptions;
private:
mutable bool exceptions_;
};
/// \brief Disable exceptions in an object derived from
/// OptionalExceptions.
///
/// This class was designed to be created on the stack, taking a
/// reference to a subclass of OptionalExceptions. (We call that our
/// "associate" object.) On creation, we save that object's current
/// exception state, and disable exceptions. On destruction, we restore
/// our associate's previous state.
class MYSQLPP_EXPORT NoExceptions
{
public:
/// \brief Constructor
///
/// Takes a reference to an OptionalExceptions derivative,
/// saves that object's current exception state, and disables
/// exceptions.
NoExceptions(const OptionalExceptions& a) :
assoc_(a),
exceptions_were_enabled_(a.throw_exceptions())
{
assoc_.disable_exceptions();
}
#if __cplusplus >= 201103L
// C++11+ alternative to the hidden copy ctor and operator= below.
NoExceptions(const NoExceptions&) = delete;
NoExceptions& operator=(const NoExceptions&) = delete;
#endif
/// \brief Destructor
///
/// Restores our associate object's previous exception state.
~NoExceptions()
{
assoc_.set_exceptions(exceptions_were_enabled_);
}
private:
const OptionalExceptions& assoc_;
bool exceptions_were_enabled_;
#if __cplusplus < 201103L
// Hidden assignment operator and copy ctor, because we should not
// be copied.
NoExceptions(const NoExceptions&);
NoExceptions& operator=(const NoExceptions&);
#endif
};
} // end namespace mysqlpp
#endif // MYSQLPP_NOEXCEPTIONS_H