1

在我的一个项目中,我用 const char 初始化了一个 const std::string& (在编译时知道),打印的结果令人惊讶。

我在 Ubuntu 12.04x32 上使用 gcc 版本 4.6.3 (Ubuntu/Linaro 4.6.3-1ubuntu5)

示例:

#include <string>
#include <iostream>
using namespace std;

class A
{
    public:
        A() : str("test"){};
    const string& str;
};

int main(int argc,char* argv[])
{ 
  A a;
  cout<<a.str<<endl;
  return 0;
};

所以,正常情况下,它会显示“测试”,但是,这是我输出的一部分:

testP�����utf8)����lower_case_table_names�xW�xW� ��     ����127.0.0.1utf8W�Y127.0.0.1 via TCP/IP127.0.0.1
        �5.5.31-0ubuntu0.12.04.1rootW�rootW�1����metadata_use_info_schema!P�XP�� VARaءpW�P4:����SHOW SESSION VARIABLES LIKE 'lower_case_table_names'`(���TCP/IP (3)@!K!K!�
.....
<charset name="gb2312">
  <family>Simplified Chinese</family>
  <description>GB2312 Simplified Chinese</description>
  <alias>chinese</alias>
  <alias>iso-ir-58</alias>
   <collation name="gb2312_chinese_ci"  id="24" order="Chinese">
    <flag>primary</flag>
   <flag>compiled</flag>
  </collation>
  <collation name="gb2312_bin"  id="86">
    <flag>binary</flag>
    <flag>compiled</flag>
  </collation>
</charset>

输出连续 1000 行。

所以我的问题很简单:为什么?(我尝试在“测试”的末尾添加一个 \0,但没有改变)

4

3 回答 3

11

那是因为您正在取消引用一个悬空引用,这会给您的程序带来未定义的行为。根据 C++11 标准的第 12.2/5 段:

[...] 临时绑定到构造函数的 ctor-initializer (12.6.2) 中的引用成员,直到构造函数退出。[...]

这意味着构造函数的初始化列表中绑定的临时std::string对象str将在您取消引用时失效str

你有两种方法可以解决这个问题。要么使您的数据成员stra const char*,要么使其成为std::string(不是引用类型)。我个人会选择最后一个选项:

class A
{
public:
    A() : str("test") { }
    std::string str;
};

另请注意,函数定义后不需要分号。

于 2013-05-23T13:02:05.227 回答
2

初始化 astring &需要 astring并且您正在提供 a const char *。当您尝试使用该引用时,编译器会生成一个临时的string,初始化string &然后销毁string未定义的行为。

于 2013-05-23T13:02:12.580 回答
1

您将引用绑定到构造函数退出时消失str的临时对象。std::string

于 2013-05-23T13:02:08.823 回答