SoPlex Doxygen Documentation
spxout.h
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 /**@file spxout.h
17  * @brief Wrapper for different output streams and verbosity levels.
18  */
19 #ifndef _SPXOUT_H_
20 #define _SPXOUT_H_
21 
22 #include <iostream>
23 #include <iomanip>
24 #include "spxdefines.h"
25 
26 // ----------------------------------------------------------------------
27 // class SPxOut
28 // ----------------------------------------------------------------------
29 
30 namespace soplex
31 {
32 
33 /**@class SPxOut
34  @ingroup Elementary
35 
36  @brief Wrapper for several output streams.
37  A verbosity level is used to decide which stream to use and whether to
38  really print a given message. Regardless of whether the verbosity level
39  is set via a manipulator or via the member function, it is persistent
40  until a new value is set.
41 
42  Most ostream member functions are not provided here; use the corresponding
43  stream manipulators (e.g., @c setprecision()) instead. These are passed on
44  to the <em>current</em> ostream, which is chosen according to the verbosity
45  level. In particular, this means that the first element in an output stream
46  should always be the verbosity. For instance, use
47  @code
48  spxout << verb( SPxOut::WARNING ) << std::setw( 15 ) << 42 << std::endl;
49  @endcode
50  or
51  @code
52  spxout.setVerbosity( SPxOut::WARNING );
53  spxout << std::setw( 15 ) << 42 << std::endl;
54  @endcode
55  instead of
56  @code
57  spxout << std::setw( 15 ) << verb( SPxOut::WARNING ) << 42 << std::endl;
58  @endcode
59  in order to make sure that @c std::setw( 15 ) is applied to the warning stream.
60 */
61 class SPxOut
62 {
63 public:
64 
65 
66  //-----------------------------------
67  /**@name Output control types */
68  //@{
69  /// Verbosity level
70  typedef enum
71  {
72  // Note: the implementation uses the fact that ERROR == 0
73  // and that the verbosity levels are subsequent numbers.
74  // If you change this, change the implementation as well.
75  ERROR = 0,
76  WARNING = 1,
77  DEBUG = 2,
78  INFO1 = 3,
79  INFO2 = 4,
80  INFO3 = 5
81  } Verbosity;
82 
83  /// helper struct for the output operator
85  {
86  /// verbosity level
88  };
89  //@}
90 
91  //-----------------------------------
92  /**@name Construction / destruction */
93  //@{
94  /// constructor
95  SPxOut();
96  /// destructor
97  virtual ~SPxOut();
98  //@}
99 
100  //-----------------------------------
101  /**@name Verbosity */
102  //@{
103  ///
104  virtual void
106  {
107  m_verbosity = v;
108  }
109  ///
110  inline Verbosity
112  const
113  {
114  return m_verbosity;
115  }
116  //@}
117 
118  //----------------------------------------
119  /**@name Wrappers for the current stream */
120  //@{
121  ///
122  inline bool good() const
123  {
124  return getCurrentStream().good();
125  }
126  ///
127  inline bool operator ! () const
128  {
129  return ! getCurrentStream();
130  }
131  ///
132  inline std::streamsize precision() const
133  {
134  return getCurrentStream().precision();
135  }
136  //@}
137 
138  //-----------------------------------
139  /**@name Getting / setting streams */
140  //@{
141  /// Sets the stream for the specified verbosity level.
142  virtual void
143  setStream( const Verbosity& verbosity,
144  std::ostream& stream )
145  {
146  m_streams[ verbosity ] = &stream;
147  }
148  /// Returns the stream for the specified verbosity level.
149  inline std::ostream&
150  getStream( const Verbosity& verbosity )
151  const
152  {
153  return *(m_streams[ verbosity ]);
154  }
155  /// Returns the stream for the current verbosity.
156  inline std::ostream&
158  const
159  {
160  return getStream( getVerbosity() );
161  }
162  //@}
163 
164 private:
165 
166  //-----------------------------------
167  /**@name Private data */
168  //@{
169  /// verbosity level
171  /// array of pointers to internal streams, indexed by verbosity level
172  std::ostream** m_streams;
173  //@}
174 
175  //-----------------------------------
176  /**@name Blocked */
177  //@{
178  /// copy constructor
179  SPxOut( const SPxOut& );
180  /// assignment operator
181  SPxOut& operator=( const SPxOut& );
182  //@}
183 };
184 
185  // ---------------------------------------------------------
186  // Manipulators
187  // ---------------------------------------------------------
188 
189 
190  //-------------------------------------------
191  /**@name Verbosity manipulator
192  Manipulators are implemented in a similar way as done for @c setw(),
193  @c setprecision(), etc. in the standard library file iomanip. For
194  instance, the non-member function \ref verb() "verb(v)" returns a
195  struct struct_Severity which contains only the verbosity level.
196  Calling
197  @code
198  SPxOut spxout;
199  spxout << verb( SPxOut::ERROR ) << "This is an error!" << std::endl;
200  @endcode
201  passes such a struct to the output operator defined below, which
202  extracts the verbosity level from the struct and passes it to the
203  member function SPxOut::setVerbosity().
204  */
205  //@{
206  /// manipulator to be used in an output statement
209  {
210  SPxOut::struct_Verbosity verbosity;
211  verbosity.v_ = v;
212  return verbosity;
213  }
214 
215  /// output operator with verbosity level struct
216  inline SPxOut&
217  operator<< ( SPxOut& stream,
218  const SPxOut::struct_Verbosity& verbosity )
219  {
220  stream.setVerbosity( verbosity.v_ );
221  return stream;
222  }
223  //@}
224 
225  //--------------------------------------------------------
226  /**@name Output of standard manipulators and other types
227  *
228  * We have to define an output operator for many kinds of numeric
229  * types here because they can all be more or less casted into each
230  * other. When using only a template type, it is not clear what the
231  * compiler makes out of it (according to lint).
232  */
233  //@{
234  ///
235 #define PASS_TO_CURRENT_OSTREAM( t ) \
236  if ( _spxout.getVerbosity() <= Param::verbose() ) \
237  _spxout.getCurrentStream() << t; \
238  return _spxout;
239 
240  /// Passes instances of type \p Type to the current stream.
241 #define DEFINE_OUTPUT_OPERATOR( Type ) \
242  inline SPxOut& \
243  operator<< ( SPxOut& _spxout, Type t ) \
244  { PASS_TO_CURRENT_OSTREAM( t ) }
245 
246  DEFINE_OUTPUT_OPERATOR( long )
247  DEFINE_OUTPUT_OPERATOR( unsigned long )
248  DEFINE_OUTPUT_OPERATOR( bool )
249  DEFINE_OUTPUT_OPERATOR( short )
250  DEFINE_OUTPUT_OPERATOR( unsigned short )
252  DEFINE_OUTPUT_OPERATOR( unsigned int )
253  DEFINE_OUTPUT_OPERATOR( double )
254  DEFINE_OUTPUT_OPERATOR( float )
255  DEFINE_OUTPUT_OPERATOR( long double )
256  DEFINE_OUTPUT_OPERATOR( const void* )
257 
258  /// Passes standard manipulators without arguments, like @c std::endl
259  /// or @c std::ios::right to the current stream.
260  inline SPxOut&
261  operator<< ( SPxOut& _spxout,
262  std::ostream& (*manip)( std::ostream& ) )
263  { PASS_TO_CURRENT_OSTREAM( manip ) }
264 
265  //lint -e{818} (pointer could be made const; this is ok.)
266  /// Passes everything else to the current stream. In particular,
267  /// this includes structs corresponding to manipulators with arguments,
268  /// such as the struct @c _Setw for the @c setw() manipulator.
269  template< typename T >
270  inline SPxOut&
271  operator<< ( SPxOut& _spxout, T t )
272  { PASS_TO_CURRENT_OSTREAM( t ) }
273  //@}
274 
275  // ---------------------------------------------------------
276  // Declare a global SPxOut instance
277  // ---------------------------------------------------------
278 
279  //-----------------------------------
280  /**@name Global instance */
281  //@{
282  ///
283  extern SPxOut spxout;
284  //@}
285 
286 } // namespace soplex
287 
288 
289 #endif // _SPXOUT_H_
290 
291 //-----------------------------------------------------------------------------
292 //Emacs Local Variables:
293 //Emacs mode:c++
294 //Emacs c-basic-offset:3
295 //Emacs tab-width:8
296 //Emacs indent-tabs-mode:nil
297 //Emacs End:
298 //-----------------------------------------------------------------------------
299