Я хочу расширить поведение объекта basic_streambuf с помощью шаблона декоратора.Вот что я сейчас получил:
template<typename char_type, class traits_type>
class forwarding_basic_streambuf
: boost::noncopyable,
public std::basic_streambuf<char_type, traits_type>
{
public:
typedef std::basic_streambuf<char_type, traits_type> forwarded_type;
forwarding_basic_streambuf(forwarded_type& fwd_buf)
: m_fwd(&fwd_buf) { }
virtual ~forwarding_basic_streambuf() { }
// locales:
// std::locale pubimbue(std::locale const& loc);
// => Calls: imbue(loc) | Returns: Previous value of getloc();
// std::locale getloc () const;
// => Returns: If pubimbue() has ever been called, then the last value of loc supplied, otherwise the
// current global locale, locale(), in effect at the time of construction. If called after
// pubimbue() has been called but before pubimbue has returned (i.e., from within the call
// of imbue()) then it returns the previous value.
// buffer management and positioning:
// forwarded_type* pubsetbuf (char_type* s, std::streamsize n); => Returns: setbuf(s, n)
// pos_type pubseekoff(off_type off, std::ios_base::seekdir way,
// std::ios_base::openmode which = std::ios_base::in | std::ios_base::out);
// => Returns seekoff(off, way, which)
// pos_type pubseekpos(pos_type sp,
// std::ios_base::openmode which = std::ios_base::in | std::ios_base::out);
// => Returns: seekpos(sp, which)
// int pubsync (); => Returns: sync()
// get and put areas:
// get area:
// std::streamsize sgetn (char_type* s, std::streamsize n); => Returns: xsgetn(s, n)
// put area:
// std::streamsize sputn(char_type const* s, std::streamsize n); => Returns: xsputn(s, n)
protected:
// virtual functions:
// locales:
virtual void imbue(std::locale const& loc) { this->m_fwd->pubimbue(loc); }
// buffer management and positioning:
virtual forwarded_type* setbuf (char_type* s, std::streamsize n)
{ return this->m_fwd->pubsetbuf(s, n); }
virtual pos_type seekoff(off_type off, std::ios_base::seekdir way,
std::ios_base::openmode which = std::ios_base::in | std::ios_base::out)
{ return this->m_fwd->pubseekoff(off, way); }
virtual pos_type seekpos(pos_type sp,
std::ios_base::openmode which = std::ios_base::in | std::ios_base::out)
{ return this->m_fwd->pubseekpos(sp, which); }
virtual int sync ()
{ return this->m_fwd->pubsync(); }
// get and put areas:
// get area:
virtual std::streamsize xsgetn(char_type* s, std::streamsize n)
{ return this->m_fwd->sgetn(s, n); }
virtual int_type uflow()
{
if (traits_type::eq_int_type(this->underflow(), traits_type::eof()))
return traits_type::eof();
return this->m_fwd->sgetc();
}
// put area:
virtual std::streamsize xsputn (char_type const* s, std::streamsize n)
{ return this->m_fwd->sputn(s, n); }
virtual int_type overflow(int_type c = traits_type::eof())
{
if (traits_type::eq_int_type(c, traits_type::eof()))
return traits_type::not_eof(c);
return this->m_fwd->sputc(traits_type::to_char_type(c));
}
private:
forwarded_type* m_fwd;
};
Основная цель (в качестве первого шага) состоит в том, чтобы просто перенаправить каждую функциональность на оформленный объект.Таким образом, должна быть возможность использовать этот декоратор даже с указателем на его базовый класс.
Все это прекрасно работает для написания методов, но я не знаю, как обращаться с функцией underflow (), который вызывается из uflow () и sgetc ().