cout一个字符串流,但打印一个指针

cout a stringstream but print a pointer

本文关键字:一个 打印 指针 cout 字符串      更新时间:2023-10-16

这是我的代码:

#include <iostream>
#include <sstream>
#include <string>
using namespace std;
class TestLog: public std::stringstream
{
public:
~TestLog()
{
cout << (str()) << endl; // Why does this print an address ?
}
};
int main()
{
TestLog() << "Hello World!"; //test 1 print an address
stringstream ss;
ss << "Hello World!";
cout << (ss.str()) << endl; //test 2 print a string
return 0;
}

输出:

0x401b90

你好,世界!

编译器信息:

g++(GCC)4.8.5 20150623(Red Hat 4.8.5-11)

在我看来,(a) std::stringstream的str()方法返回一个字符串。(b) std::cout是std::ostream的对象。因此,两个测试都将调用ostream的相同运算符函数,并打印相同的";你好世界";。但是测试1打印地址,测试2打印正确的"地址";你好世界">
我怎么了?谢谢

ss << "Hello World!";被解析为对以下重载的调用(本页上的#2):

template <class Traits>
std::basic_ostream<char, Traits> &std::operator << (
std::basic_ostream<char, Traits> &os, char const *s
);

这个重载将字符串文字衰减为char const *,然后打印它

要搅拌泥浆,我们可以尝试以下片段:

TestLog tl;
tl << "Hello World!";

这将解决上面的过载,并打印Hello World!。这是因为tl是一个左值,它可以绑定到作为非const左值引用的第一个参数。

在您的示例中,TestLog()是一个右值——此重载不匹配!因此,选择了另一个过载(此处为#7):

std::basic_ostream &std::basic_ostream::operator << (void const *value);

这是一个成员函数重载,从std::stringstream继承而来。即使不能将非const引用绑定到右值,也可以调用右值上的非const成员函数。因此,这个重载是一个有效的匹配,并且它是被选择的——打印文本的地址,就像它是任何旧指针一样。

C++11带来了一个新的过载来解决这个问题,在这里的#3可见:

template <class CharT, class Traits, class T>
std::basic_ostream<CharT, Traits> &operator << (
basic_ostream<CharT, Traits> &&os, T const &value
);

T const &与文字的char const[N]类型完全匹配,因此它的排名高于void const *重载。作为第一个参数的右值引用绑定到临时的just fine。

命名的右值引用被认为是左值,因此该函数可以再次调用os << value;,以蹦床返回到左值流的过载集。因此,在C++11及以上版本中,两行都打印Hello World!

相关文章: