Standard no-op output stream

后端 未结 6 1802
余生分开走
余生分开走 2020-11-29 03:25

Is there a way to create an ostream instance which basically doesn\'t do anything ?

For example :

std::ostream dummyStream(...);
dummyStream <<         


        
相关标签:
6条回答
  • 2020-11-29 03:41

    If this is to disable logging output, your dummyStream would still cause arguments to be evaluated. If you want to minimize impact when logging is disabled, you can rely on a conditional, such as:

    #define debugStream \
        if (debug_disabled) {} \
        else std::cerr
    

    So if you have code like:

    debugStream << "debugging output: " << foo() << std::endl;
    

    No arguments will be evaluated if debug_disabled is true.

    0 讨论(0)
  • 2020-11-29 03:47

    The basic method voor new stream classes is:

    1. Derive a class from std::streambuf;
    2. Override the virtual functions in that class. This is where the real work is done. In your case, empty implementations should be good enough.
    3. Derive a class from std::ostream with one member, your streambuf class.
    4. The constructor of your streamclass should forward the pointer to that member to the base constructor of std::ostream.

    I'm afraid you won't get rid of the formatting step, though.

    Hopefully this gives you some pointers; I don't have the time to expand this into a full answer, sorry.

    Update: See john's answer for details.

    0 讨论(0)
  • 2020-11-29 03:52

    If you are concerned about the overhead of your debugger then you can write a very simple code to void out your debug messages on compilation. This is what I use for my c++ programs.

    #include <iostream>
    #define DEBUGGING // Define this in your config.h or not.
    #ifdef DEBUGGING
    /*
     * replace std::cout with your stream , you don't need to
     * worry about the context since macros are simply search
     * and replace on compilation.
     */
    #define LOG_START std::cout <<
    #define LOG_REDIR <<
    #define LOG_END   << std::endl;
    #else
    #define LOG_START (void)
    #define LOG_REDIR ;(void)
    #define LOG_END   ;
    #endif // DEBUGGING
    
    int main(){
    LOG_START "This is a log message " LOG_REDIR "Still a log message." LOG_END;
    return 0;
    }
    

    Now when making your project , check if the user wants to disable the logging , if so , just undefine the DEBUGGING macro or whatever macro you choose to check for.

    Now your code will be optimized by the compiler , Because when anything is voided , it will not be included in the resulting binary(most of the time) , making the binary production ready.

    0 讨论(0)
  • 2020-11-29 03:56

    I needed a null stream that was of type ostream so I did something like this:

    struct NullStream: public stringstream {
       NullStream(): stringstream() {}
    };
    
    template<typename T>
    void operator<<(const NullStream&, const T&) {}
    

    Application code:

    NullStream ns;
    ostream &os = ns;
    os << "foo";
    

    The real issue is all the public methods that I inherited but don't care about so I just didn't bother overriding them.

    0 讨论(0)
  • 2020-11-29 04:00

    You need a custom streambuf.

    class NullBuffer : public std::streambuf
    {
    public:
      int overflow(int c) { return c; }
    };
    

    You can then use this buffer in any ostream class

    NullBuffer null_buffer;
    std::ostream null_stream(&null_buffer);
    null_stream << "Nothing will be printed";
    

    streambuf::overflow is the function called when the buffer has to output data to the actual destination of the stream. The NullBuffer class above does nothing when overflow is called so any stream using it will not produce any output.

    0 讨论(0)
  • 2020-11-29 04:04

    For runtime-controllable redirection of log messages, a self-contained solution combining the ideas of john and Sjoerd:

    class DebugStream {
    private:
        class NullStream : public std::ostream {
        private:
            class NullBuffer : public std::streambuf {
            public:
                int overflow(int c) override { return c; }
            } buffer_;
        public:
            NullStream() : std::ostream(&buffer_) {}
        } null_;
    
        std::ostream &output_;
        bool enabled_;
    
    public:
        DebugStream(std::ostream &output = std::cout) : output_(output), enabled_(false) {}
        void enable(const bool enable) { enabled_ = enable; }
    
        template <typename T> std::ostream& operator<<(const T &arg) {
            if (enabled_) return output_ << arg;
            else return null_ << arg;
        }
    };
    
    extern DebugStream debug_stream;
    #define TRACE_ENABLE(x) debug_stream.enable(x)
    #define TRACELN(x) debug_stream << x << std::endl
    #define TRACE(x) debug_stream << x
    

    Then you can do stuff like:

    TRACELN("The value of x is " << x " and the value of y is " << y);
    

    It would also be easy to just remove the trace statements from a release version completely with #define the trace macros to empty statements.

    You still need to define debug_stream somewhere global, though.

    0 讨论(0)
提交回复
热议问题