0

我有一个字符串向量,它们无缘无故地改变了它的内容。无法真正解释发生了什么。很抱歉列出了很长的清单,但这真的让我很烦。

我有一个 GUI 应用程序,它加载一些文件并使用可以通过使用parse(int argc, char* argv[])方法设置状态的读取器对象。通过选中各种框并输入值,可以在对话框中设置参数。这是我用来保存对话框数据的结构:

struct PointFilter
{
    PointFilter(): argc(0) {};
    ~PointFilter() {};
    int argc;
    std::vector<std::string> args;
};

该结构是对话框类的成员,在按下确定按钮后,它会填充适当的值。这些值从对话框上的文本框中获取到字符串流中,然后推回 std::vector:

class AdvancedLoadDialog
{
public:
    AdvancedLoadDialog(const Glib::RefPtr<Gtk::Builder>&);
    ~AdvancedLoadDialog();

    PointFilter get_point_filter()
    {
        return this->point_filter;
    }

private:
    PointFilter point_filter;
    void on_ok_btn_clicked();

void AdvancedLoadDialog::on_ok_btn_clicked()
{
    std::stringstream filter_stream;
        // filter_stream << some_values_from_textboxes ...

    std::vector<std::string> args;
    std::string arg;

    // we need a dummy first argument to emulate the command line
    args.push_back("filter");

    while (filter_stream >> arg)
    {
        args.push_back(arg);
    }       

    point_filter.argc = args.size() > 1 ? args.size() : 0;
    point_filter.args = args;

    advanced_load_dialog->hide_all();
}

到目前为止一切正常,我们有一个AdvancedLoadDialog带有point_filter成员的对象来保存我们的论点。现在在一个单独的窗口中,我获取 point_filter 对象并将其传递给LoadWorker类的构造函数,该构造函数加载文件并且还有一个PointFilter成员。

load_worker = new LoadWorker(..., advanced_load_dialog->get_point_filter())

进而:

LoadWorker::LoadWorker(..., PointFilter pf) :
    point_filter    (pf)

一切都很好。现在在LoadWorker::run()函数中,我从 point_filter 获取参数,将它们转换为 std::vector 并将它们传递给我需要的 `parse(int argc, char* argv[]) 函数。

void LoadWorker::run()
{
    std::cout << "LoadWorker::file_filter contents: \n"
            << "point_filter.argc: " << point_filter.argc << "\n"
            << "point_filter.args: " << std::endl;
    for (int i = 0; i < point_filter.argc; ++i)
        std::cout << point_filter.args[i] << std::endl;

        // ...

    if (point_filter.argc != 0)
    {
        std::cout << "Using filter: " << std::endl;

        std::vector<char*> argv;

        for (std::vector<std::string>::const_iterator it = point_filter.args.begin();
                it != point_filter.args.end(); ++it)
        {
            argv.push_back(const_cast<char*>(it->c_str()));
        }
        argv.push_back(0);

        for (int i = 0; i < point_filter.argc; ++i)
        {
            std::cout << argv[i] << std::endl;
        }

        if (!lasreadopener.parse(point_filter.argc, &argv[0]))
        {
            send_message("Error parsing filter parameters.");
            sig_fail();
            return;
        }
    }
}  

现在这个工作......一次。您会注意到参数被打印了两次,首先是作为LoadWorker::point_filter.argsvector 的元素,然后是作为vector<char*> argv. 如果我设置过滤器,然后按下加载按钮,一切正常。如果我然后尝试加载另一个文件,AdvancedLoadDialog::point_filter根本不改变,参数就会消失。这是一个尝试连续加载两个文件的示例输出。

LoadWorker::file_filter 内容: point_filter.argc: 6 point_filter.args: filter -clip_z_above 12 -keep_intensity 11 222 使用过滤器: filter -clip_z_above 12 -keep_intensity 11 222 LoadWorker::file_filter 内容: point_filter.argc: 6 point_filter.args: filter clip_z_above 2 keep_intensity 1 22 使用过滤器:过滤器

// 这里有 6 个空行

更奇怪的是,在第二次运行期间,除了第一个字符串之外的每个字符串都point_filter.args缺少第一个字符,并且argv它们都是空的。

有什么线索吗?

4

2 回答 2

1

std::string::c_str在做涉及到你在这里做的事情时要非常非常谨慎:

argv.push_back(const_cast<char*>(it->c_str()));

如果要保留 提供的值c_str,应先将它们复制到新容器中。

如果要信任cplusplus.com(并非总是如此),

此数组中的值不应在程序中修改,仅保证在下一次调用字符串对象的非常量成员函数之前保持不变

虽然我不能立即从其他来源获得任何更有用的信息。

于 2012-06-27T12:15:04.730 回答
1

在 AdvancedLoadDialog 中,尝试推送分配的 arg 版本而不是自动版本。您似乎正在推送对在离开 AdvancedDialogLoad 时将被破坏的变量的引用。

于 2012-06-27T12:36:45.990 回答