C ++自定义流操纵器,可更改流中的下一项

在C ++中,要打印十六进制数字,请执行以下操作:


int num = 10;

std::cout << std::hex << num; // => 'a'

我知道我可以创建一个仅向流中添加内容的操纵器,如下所示:


std::ostream& windows_feed(std::ostream& out)

{

    out << "\r\n";

    return out;

}


std::cout << "Hello" << windows_feed; // => "Hello\r\n"

但是,如何创建一个类似于“ hex”的操纵器来修改要出现在流中的项目?作为一个简单的示例,我将如何在此处创建plusone机械手?:


int num2 = 1;

std::cout << "1 + 1 = " << plusone << num2; // => "1 + 1 = 2"


// note that the value stored in num2 does not change, just its display above.

std::cout << num2; // => "1"


临摹微笑
浏览 457回答 3
3回答

侃侃无极

首先,您必须将一些状态存储到每个流中。您可以使用函数iword和传递给它的索引来实现,该函数由给出xalloc:inline int geti() {&nbsp;&nbsp; &nbsp; static int i = ios_base::xalloc();&nbsp; &nbsp; return i;}ostream& add_one(ostream& os) { os.iword(geti()) = 1; return os; }&nbsp;ostream& add_none(ostream& os) { os.iword(geti()) = 0; return os; }有了适当的设置,您已经可以在所有流中检索某些状态。现在,您只需要加入相应的输出操作即可。数值输出由构面完成,因为它可能取决于语言环境。所以你可以做struct my_num_put : num_put<char> {&nbsp; &nbsp; iter_type&nbsp;&nbsp; &nbsp; do_put(iter_type s, ios_base& f, char_type fill, long v) const {&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; return num_put<char>::do_put(s, f, fill, v + f.iword(geti()));&nbsp;&nbsp; &nbsp; }&nbsp;&nbsp; &nbsp; iter_type&nbsp;&nbsp; &nbsp; do_put(iter_type s, ios_base& f, char_type fill, unsigned long v) const {&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; return num_put<char>::do_put(s, f, fill, v + f.iword(geti()));&nbsp;&nbsp; &nbsp; }&nbsp;};&nbsp;现在,您可以测试这些东西了。int main() {&nbsp; &nbsp; // outputs: 11121011&nbsp; &nbsp; cout.imbue(locale(locale(),new my_num_put));&nbsp; &nbsp; cout << add_one << 10 << 11&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<< add_none << 10 << 11;}如果您只想增加下一个数字,只需0在每次调用后将单词再次设置为do_put。

SMILET

我完全同意Neil Butterworth的观点,但是在特定情况下,您使用的是这种完全骇人听闻的技巧。不要在任何生产代码中执行此操作。它有很多错误。一方面,它仅适用于您的单行代码,而不会更改基础流的状态。class plusone_stream : public std::ostream{&nbsp; public:&nbsp; &nbsp; std::ostream operator<<(int i)&nbsp; &nbsp; {&nbsp; &nbsp; &nbsp; _out << i+1;&nbsp; &nbsp; &nbsp; return *this;&nbsp; &nbsp; }};std::ostream& plusone(std::ostream& out){&nbsp; &nbsp; return plusone_stream(out);}
打开App,查看更多内容
随时随地看视频慕课网APP