實作一個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;
}