How to properly overload the << operator for an ostream?

前端 未结 5 1200
佛祖请我去吃肉
佛祖请我去吃肉 2020-11-22 04:54

I am writing a small matrix library in C++ for matrix operations. However my compiler complains, where before it did not. This code was left on a shelf for 6 months and in b

相关标签:
5条回答
  • 2020-11-22 05:17

    Just telling you about one other possibility: I like using friend definitions for that:

    namespace Math
    {
        class Matrix
        {
        public:
    
            [...]
    
            friend std::ostream& operator<< (std::ostream& stream, const Matrix& matrix) {
                [...]
            }
        };
    }
    

    The function will be automatically targeted into the surrounding namespace Math (even though its definition appears within the scope of that class) but will not be visible unless you call operator<< with a Matrix object which will make argument dependent lookup find that operator definition. That can sometimes help with ambiguous calls, since it's invisible for argument types other than Matrix. When writing its definition, you can also refer directly to names defined in Matrix and to Matrix itself, without qualifying the name with some possibly long prefix and providing template parameters like Math::Matrix<TypeA, N>.

    0 讨论(0)
  • 2020-11-22 05:24

    Assuming that we're talking about overloading operator << for all classes derived from std::ostream to handle the Matrix class (and not overloading << for Matrix class), it makes more sense to declare the overload function outside the Math namespace in the header.

    Use a friend function only if the functionality cannot be achieved via the public interfaces.

    Matrix.h

    namespace Math { 
        class Matrix { 
            //...
        };  
    }
    std::ostream& operator<<(std::ostream&, const Math::Matrix&);
    

    Note that the operator overload is declared outside the namespace.

    Matrix.cpp

    using namespace Math;
    using namespace std;
    
    ostream& operator<< (ostream& os, const Matrix& obj) {
        os << obj.getXYZ() << obj.getABC() << '\n';
        return os;
    }
    

    On the other hand, if your overload function does need to be made a friend i.e. needs access to private and protected members.

    Math.h

    namespace Math {
        class Matrix {
            public:
                friend std::ostream& operator<<(std::ostream&, const Matrix&);
        };
    }
    

    You need to enclose the function definition with a namespace block instead of just using namespace Math;.

    Matrix.cpp

    using namespace Math;
    using namespace std;
    
    namespace Math {
        ostream& operator<<(ostream& os, const Matrix& obj) {
            os << obj.XYZ << obj.ABC << '\n';
            return os;
        }                 
    }
    
    0 讨论(0)
  • 2020-11-22 05:24

    In C++14 you can use the following template to print any object which has a T::print(std::ostream&)const; member.

    template<class T>
    auto operator<<(std::ostream& os, T const & t) -> decltype(t.print(os), os) 
    { 
        t.print(os); 
        return os; 
    } 
    

    In C++20 Concepts can be used.

    template<typename T>
    concept Printable = requires(std::ostream& os, T const & t) {
        { t.print(os) };
    };
    
    template<Printable T>
    std::ostream& operator<<(std::ostream& os, const T& t) { 
        t.print(os); 
        return os; 
    } 
    
    0 讨论(0)
  • 2020-11-22 05:31

    You have declared your function as friend. It's not a member of the class. You should remove Matrix:: from the implementation. friend means that the specified function (which is not a member of the class) can access private member variables. The way you implemented the function is like an instance method for Matrix class which is wrong.

    0 讨论(0)
  • 2020-11-22 05:35

    To add to Mehrdad answer ,

    namespace Math
    {
        class Matrix
        {
           public:
    
           [...]
    
    
        }   
        std::ostream& operator<< (std::ostream& stream, const Math::Matrix& matrix);
    }
    

    In your implementation

    std::ostream& operator<<(std::ostream& stream, 
                         const Math::Matrix& matrix) {
        matrix.print(stream); //assuming you define print for matrix 
        return stream;
     }
    
    0 讨论(0)
提交回复
热议问题