实现 std::basic_streambuf 子类来操作输入

Posted

技术标签:

【中文标题】实现 std::basic_streambuf 子类来操作输入【英文标题】:Implementing std::basic_streambuf subclass for manipulating input 【发布时间】:2014-01-29 01:44:37 【问题描述】:

我有一个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++ 方面有点新手。另请注意,我知道我可以完整地阅读输入内容,并在阅读后将其简单地转换为小写,但这更多是出于学习/学术目的,而不是任何实际用途。

【问题讨论】:

也许我在这里遗漏了一些东西,但为什么你们都继承自 std::streambuf,并包含一个作为成员变量。 @Borgleader 很好。它是一个 streambuf 对象的包装器。 首先,int underflow() 不带任何参数... @Cubbi:糟糕,复制粘贴错误(我刚刚复制了upper_streambuf 实现)。编辑问题,现在代码与我的实际代码匹配。 【参考方案1】:

underflow() 有你未能建立的后置条件:当你从它返回时,你需要在你的 streambuf 中有一个获取区域来保存你要返回的那个字符。

单字符缓冲区就足够了:

    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;
;

【讨论】:

认为我明白...“get”缓冲区是必需的,因为某些读取操作不会“消耗”字符,对吗? @dreamlax 损坏的后置条件 = 任何事情都有可能发生。具体来说,getline 调用 sbumpc(在我刚刚看到的实现中),当 get 区域为空时调用 uflow,它调用 underflow,如果不返回 eof,它会取消引用 get指针并返回那里的字符。

以上是关于实现 std::basic_streambuf 子类来操作输入的主要内容,如果未能解决你的问题,请参考以下文章

Android 使用handler实现线程间发送消息 (主线程 与 子线程之间)(子线程 与 子线程之间)

最长公共子串和子序列的Python实现,带图示。

Axure中继器实现子表的展开和收缩(载入时展开子表)

如何实现所有子视图的滚动

Java--最大子序列和实现

如何实现子组件向父组件传值