C++標準庫筆記:13.12.1 實作一個Output操作符

實作一個Output操作符,即重載IO操作符<<
表達式stream << object,根據語法規則,上式有兩種解釋:

  • stream.operator <<( object ),stream的成員函數,用於內建(基本,如int、char等)型別,stream已關閉對此的擴展之門(當然了,不然你往stream類裏面添加成員,不就改了源碼了)。
  • operator<<( stream, object),可以擴展的唯一方法,寫一個全局性的operator <<。

測試所用Fraction類

class Fraction
{
public:
    Fraction( int nNumerator = 0, int nDemominator = 1 )
        :m_nNumerator( nNumerator ), m_nDemominator( nDemominator )
    {}

    int Numerator() const { return m_nNumerator; }
    int Demominator() const { return m_nDemominator; }

private:
    int m_nNumerator;       //分子
    int m_nDemominator;     //分母
};

按”分子/分母“格式打印Fraction對象,可如下編寫operator <<:

inline std::ostream& operator<<( std::ostream& strm, 
                                const Fraction& f )
{
    strm << f.Numerator() << '/' << f.Demominator();
    return strm;
}

以上寫法存在兩個問題:

  • 由於直接使用ostream,對wchar_t就不能用了
  • 如果字段寬度標誌被設定,得到的結果可能和預期的不一樣。字段寬度會旅行於緊隨其後的改寫操作中,此處將施行於分子身上。如下代碼會導致輸出爲:f1: “2~~~~~~~/3”(其中~代表空格)

    Fraction f1( 2, 3 );
    std::cout << "f1: \"" << std::left << std::setw( 8 )
            << f1 << "\"" << std::endl;

下面這個版本可以同時解決以上兩問題:

template <class charT, class traits>
inline std::basic_ostream<charT, traits>&
operator << ( std::basic_ostream<charT, traits>& strm, 
             const Fraction& f )
{
    //string stream
    //with same foramt
    //without special field width
    std::basic_ostringstream<charT, traits> s;
    s.copyfmt( strm );
    s.width( 0 );

    //fill sting stream
    s << f.Numerator() << '/' << f.Demominator();

    //print string stream
    strm << s.str();

    return strm;
}
發佈了164 篇原創文章 · 獲贊 106 · 訪問量 33萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章