实作一个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;
}