在结构中分配std::字符串时出现分段错误

Segmentation fault assigning std::string in a struct

本文关键字:分段 错误 字符串 分配 std 结构      更新时间:2023-10-16

以下代码在运行时会导致分段故障,我不知道原因:

#include <cstdlib>
#include <string>
#include <iostream>

using namespace std;
struct Token
{
  int num;
  string val;
};

int main()
{
  Token* tok = (Token*) malloc (sizeof(Token));
  tok -> val = "myval";
  std::cout<<tok->val;
}

参见回溯:

0  0x00007ffff7b95d9b in std::string::assign(char const*, unsigned long) () from /usr/lib/x86_64-linux-gnu/libstdc++.so.6;
1  0x0000000000400867 in main ()

不要在C++代码中使用malloc,它很少是正确的选择。

您的选择是:

Token tok;
tok.val = "myval";
auto tok = std::make_unique<Token>(); // C++14
tok->val = "myval";
auto tok = std::unique_ptr<Token>(new Token()); // C++11
tok->val = "myval";
auto tok = std::make_shared<Token>(); // C++11, use if resource is shared
tok->val = "myval";
Token* tok = new Token();
tok->val = "myval";
delete tok;

这些选项应该足以满足大多数情况。

首选从上到下的选项:默认的方式应该是创建对象,然后是unique_ptr,再是shared_ptr,只有在绝对必要的情况下,才应该处理原始指针。

原因很简单:异常安全和内存泄漏。对象不能泄漏,不能忘记deleteunique_ptrshared_ptr,但可以使用原始指针。此外,在出现异常的情况下,原始指针永远不会被删除。unique_ptr应该优先于shared_ptr,因为shared_ptr有开销(原子计数器使其线程安全)。

演示一切编译良好(没有C++14 make_unique):演示

malloc不调用任何构造函数。尝试使用new,它应该可以正常工作。

另外,不要忘记稍后delete,或者使用智能指针或直接将变量放在堆栈上。

以下是您不能将malloc与std::string:一起使用的原因

在C++中,不能用非平凡构造函数对类进行malloc。你做了什么from malloc是一个原始内存块,它不包含正确构造的对象。任何将该内存用作"真实"对象将失败。

以及(已经在其他答案中指出):

问题是malloc没有调用example的构造函数。由于字符串通常表示为堆栈上的指针,因此设置为零,并且取消引用空指针。你需要使用新的。

如果你坚持使用malloc,你可以使用placementnew(详见链接),也可以只使用普通的ol'const char*:

struct Token
{
  int num;
  const char* val;
};
int main()
{
  Token* tok = (Token*) malloc (sizeof(Token));
  tok->val = "myval";
  std::cout << tok->val;
  free(tok);
} // will not seg fault