2

次のように、std::basic_streambufすべての出力を大文字で書き込むサブクラスがあります。

class upper_streambuf : public std::streambuf
{
    public:
        upper_streambuf(std::streambuf &real)
          : m_realBuffer(real)
        {
        }

    protected:
        virtual int overflow(int c)
        {
            int uc = std::toupper(c);
            m_realBuffer.sputc(uc);
            return uc;
        }

    private:
        std::streambuf &m_realBuffer;
};

たとえば、次のように使用します(問題なく動作するようです):

upper_streambuf buf(*std::cout.rdbuf());
std::ostream ucout(&buf);

ucout << "Hello, world!" << std::endl; // prints "HELLO, WORLD!"

私が達成したいのは、多かれ少なかれ逆です。ストリームから読み取り、すべての入力を小文字に変換したいと考えています。私は次のものを持っています:

class lower_streambuf : public std::streambuf
{
    public:
        lower_streambuf(std::streambuf &real)
          : m_realBuffer(real)
        {
        }

    protected:
        virtual int underflow()
        {
            return std::tolower(m_realBuffer.sbumpc());
        }

    private:
        std::streambuf &m_realBuffer;
};

ただし、これを次のように使用しようとすると:

lower_streambuf buf(*std::cin.rdbuf());
std::istream lcin(&buf);
std::string line;

std::getline(lcin, line);

その結果、セグメンテーション違反が発生します。間違った関数をオーバーライドまたは呼び出していますか? 私はC++の初心者であることに注意してください。また、入力を完全に読み取り、読み取り後に小文字に変換するだけでよいことは理解していますが、これは実用的なものではなく、学習/学術目的のためです。

4

1 に答える 1

4

underflow()には、確立できなかった事後条件があります。そこから戻るまでに、ストリームバッファに、返す文字を保持する get 領域が必要です。

1 文字のバッファで十分です。

    protected:
        virtual int underflow()
        {
            ch = std::tolower(m_realBuffer.sbumpc());
  +         setg(&ch, &ch, &ch+1); 
            return ch;
        }
    private:
  +     char ch; // input buffer!
        std::streambuf &m_realBuffer;
};
于 2014-01-29T02:37:58.400 に答える