1

我有一个函数,它遍历 aconst char *并使用该字符将对象添加到一个实例中,std::map如果它是一系列可识别字符之一。

#define CHARSEQ const char*

void compile(CHARSEQ s) throw (BFCompilationError)
{
     std::cout << "@Receive call " << s << std::endl;

     for(int i = 0; s[i] != '\0'; i++)
     {
         if (std::string("<>-+.,[]").find_first_of(s[i]) == std::string::npos)
         {
             throw BFCompilationError("Unknown operator",*s,i);
         }
         std::cout << "@Compiling: " << s[i] << std::endl;
         std::cout << "@address s " << (void*)s << std::endl;
         std::cout << "@var s " << s << std::endl;
         controlstack.top().push_back(opmap[s[i]]);
     }
 }

传递的字符序列是"++++++++++." 前3次迭代,print语句显示'+'、'+'和'+'的期望值,并且的值s继续为“++++++++++” +。”。但是,在第四次迭代中,它变得混乱,产生了奇怪的值,s例如'Ð''öê'、和许多其他字符序列。如果抛出异常的行被删除,并且允许循环继续,那么之后的值不会再改变。'cR ''œk's

其他函数可以访问,s但由于这不是一个多线程程序,我不明白为什么这很重要。我对为什么s会发生变化并不感到困惑,而是为什么它只在第四次迭代时发生变化。

我已经搜索过,唯一看起来完全相关的帖子就是这个,但它仍然没有回答我的问题。(研究一直很困难,因为搜索“const char* 改变值”或类似术语只会出现数百篇关于 is 的部分是 const的帖子)。

最后,我知道我可能应该使用std::string,如果没有答案,我会使用它,但我仍然想了解这种行为。

编辑:

这是调用此函数的代码。

CHARSEQ text = load(s);

std::cout << "@Receive load " << text << std::endl;

try
{
    compile(text);
}
catch(BFCompilationError& err)
{
    std::cerr << "\nError in bf code: caught BFCompilationError @" << err.getIndex() << " in file " << s << ":\n";
    std::cerr << text << '\n';
    for(int i = 0; i < err.getIndex(); i++)
    {
        std::cerr << " ";
    }
    std::cerr << "^\n";
    std::cerr << err.what() << err.getProblemChar() << std::endl;
    return 1;
}

在哪里load

CHARSEQ load(CHARSEQ fname)
{
    std::ifstream infile (fname);
    std::string data(""), line;
    if (infile.is_open())
    {
    while(infile.good())
    {
        std::getline(infile,line);
        std::cout << "@loading: "<< line << '\n';
        data += line;
    }
    infile.close();
}
else
{
    std::cerr << "Error: unable to open file: " << fname << std::endl;
}

return std::trim(data).c_str();

}

并且文件fname++++++++++.传播,使得每行有一个字符。

编辑2:

这是控制台输出的示例:

@loading: +
@loading: +
@loading: +
@loading: +
@loading: + 
@loading: +
@loading: +
@loading: +
@loading: +
@loading: +
@loading: .
@Receive load ++++++++++.
@Receive call ++++++++++.
@Compiling: +
@address s 0x7513e4
@var s ++++++++++.
@Compiling: +
@address s 0x7513e4
@var s ++++++++++.
@Compiling: +
@address s 0x7513e4
@var s ++++++++++.
@Compiling: 
@address s 0x7513e4
@var s ßu

Error in bf code: caught BFCompilationError @4 in file bf_src/Hello.txt:
ßu
^
Unknown operatorß
4

1 回答 1

8

你的load功能有缺陷。由返回的const char*指针c_str()仅在底层std::string对象存在之前有效。但是data是一个局部变量,load并且在返回后被清除。它的缓冲区不会被零覆盖,而是保留为空闲内存。因此,在返回后立即打印出值可能会起作用,但您的程序可能会在那里放置新值,并且指针指向的值会改变。

我建议使用std::string作为负载的返回值作为一种解决方法。

于 2013-01-17T22:57:00.053 回答