为啥 const char* 不能与 boost 的 stringstream 和 read_json 一起使用?

Posted

技术标签:

【中文标题】为啥 const char* 不能与 boost 的 stringstream 和 read_json 一起使用?【英文标题】:Why const char* is not working with stringstream and read_json of boost?为什么 const char* 不能与 boost 的 stringstream 和 read_json 一起使用? 【发布时间】:2015-12-04 16:05:38 【问题描述】:

我正在尝试将字符指针传递给将在内部使用字符串 read_json 的函数。但是从函数中出来后,指针的内容发生了变化。下面是代码。

string tempstr;
void someFunc()

    const char *ptr = getDiagnosticGroupValue("Device.ManagementServer");
    printf("string : %s\n", getParameterValue(ptr, "URL"));
    printf("string : %s\n", getParameterValue(ptr, "Username"));

const char* getParameterValue(const char *jsonstring, const char* param)

    return getString(jsonstring, param);

const char* getString(const char *jsonstring, const char* param)

    stringstream jsonstringstream;
    jsonstringstream << jsonstring;

    string paramstr(param);

    try
    
        boost::property_tree::ptree pt;
        boost::property_tree::read_json(jsonstringstream, pt);
        tempstr = pt.get<string>(paramstr);
        return tempstr.c_str();
    
    catch(...)
    
        jsonstringstream.clear();
        jsonstringstream.str("");
        cout << "Node Doesn't exist" << endl;
        return "N/A";
    

对于第一个节点 (URL),它返回正确的值,我可以在 someFunc() 中打印字符串。但是 ptr 内容正在被修改(它不再是 json 格式字符串)。所以第二次,当 getParameterValue() 被调用时,read_json 抛出异常,说不是 json 格式。我想知道为什么即使我声明为 const char* ptr,ptr 的内容也会发生变化。任何人都可以让我知道如何解决这个问题。

我确认问题不在于临时对象。我通过创建全局字符串并从全局字符串返回 char * 来修改代码。但我面临同样的问题。还有一件事是第一次调用getParameterValue()时,我把print语句放在read_json()之后检查ptr的值,而此时ptr已经损坏了。

getDiagnosticGrouapValue() 的实现如下。

const char* getDiagnosticGroupValue(const char* group)

    VZUAL& vzual = VZUAL::getInstance();

    string groupStr(group, group + strlen(group));
    return vzual.Diagnostics.getGroupValue(groupStr).c_str();


const string PlatformDiagnostics::getGroupValue(const string& _groupName_, const string& DUMMY)

    boost::property_tree::ptree ptRead;
    boost::property_tree::ptree ptWrite;
    string path = _groupName_ + ".";
    stringstream ss;

    try 
        boost::property_tree::read_json(Resource::diagnosticsFile.c_str(), ptRead);
    
    catch (boost::property_tree::json_parser::json_parser_error &je) 
        return Resource::defaultValue;
    

    try 
        BOOST_FOREACH(boost::property_tree::ptree::value_type &v, ptRead.get_child(path))
        ptWrite.put(v.first.data(), v.second.data());
    
    catch (...) 
    return Resource::defaultValue;
    

    boost::property_tree::write_json(ss, ptWrite);
    return ss.str();

【问题讨论】:

请不要用c标记这个问题。 【参考方案1】:

您正在返回指向在 getString 函数内部堆栈上创建的字符串的指针。这是未定义的行为。

从您的代码中,所有变量:

string paramstr(param);
boost::property_tree::ptree pt;
boost::property_tree::read_json(jsonstringstream, pt);  
return (pt.get<string>(paramstr)).c_str();

表示该字符串不是静态或全局创建的。

如果您要返回指向 jsonstring 某些部分的指针,十个也许您应该在 jsonstring 中找到该字符串并返回指向它的指针。问题是它不会以空值终止(因此您还需要返回长度)。

其他解决方案是为 getString 提供一个 char 数组,该数组将从 getString 返回。但是你必须确保它足够大,如果太小则返回NULL,以便客户端代码可以提供更大的缓冲区。

【讨论】:

我确认问题不在于临时对象。我通过创建全局字符串并从全局字符串返回 char * 来修改代码。但我面临同样的问题。还有一件事是第一次调用getParameterValue()时,我把print语句放在read_json()后面来检查ptr的值,此时ptr已经损坏了。 @kadina 它是 const char * 所以很难相信它被修改了,你也把它放在 stringstream 里面,这肯定不会改变它。也许这个json中没有Usernameproblem is not with temporary object - 这不是真的,你只是有不止一个问题:-) 你当前的全局变量解决方案是可以的,直到你尝试从多个线程中使用你的函数,或者尝试将返回值存储在某个地方。 用户名存在于 json 字符串中。我使用参数“用户名”删除了对 getString() 的调用,而是在 someFunc() 中的 getString() 之后打印了 ptr,它正在打印损坏的字符串。 我的想法已经不多了。我仍然不确定如何调试此问题以及如何修改 const char* ptr。 你能添加你的 getDiagnosticGroupValue 实现吗?

以上是关于为啥 const char* 不能与 boost 的 stringstream 和 read_json 一起使用?的主要内容,如果未能解决你的问题,请参考以下文章

为啥 boost::interprocess::managed_shared_ptr to non-const 不能转换为 managed_shared_ptr to const

为啥在片段中从 char* 转换为 std::string 比转换为 const char* 更可取?

为啥 atoi 函数不能将 const char * 转换为 int?

将标记转换为 char* const* 时,使用 boost 标记字符串失败

如何定义 boost tokenizer 以返回 boost::iterator_range<const char*>

C++ boost.python 无法将 const char* 转换为 str