【问题标题】:free strdup output in a loop在循环中释放 strdup 输出
【发布时间】:2018-05-14 15:25:55
【问题描述】:

我有一个循环尝试从 qml 列表对象中读取数据,这是我的循环

char * argvarry[(gcps.size() * 5) + 8];
argvarry[0] = "-of";
argvarry[1] = "GTiff";
argvarry[2] = "-a_nodata";
argvarry[3] = "'0 0 0'";
argvarry[4] = "-a_srs";
argvarry[5] = a_srs;

int argc = 6;

for (int i = 0;i < gcps.size(); i++) {
    argvarry[argc] = "-gcp";argc++;//gcp_values
    gcppoint_ *a = qobject_cast<gcppoint_ *>(gcps.at(i).value<QObject *>());

      argvarry[argc]= strdup( const_cast<char*>(QString::number(a->row()).toStdString().c_str())); argc++;
      argvarry[argc] = strdup(const_cast<char*>(QString::number(a->column()).toStdString().c_str())); argc++;
      argvarry[argc] = strdup(const_cast<char*>(QString::number(a->lon()).toStdString().c_str())); argc++;
      argvarry[argc] =strdup( const_cast<char*>(QString::number(a->lat()).toStdString().c_str())); argc++;
}

好吧,我尝试了一些方法来使上述代码工作,但没有工作, 例如,如果我使用strdup,它可以工作,但需要释放,我尝试将strdup 的输出保存在一个变量中,然后在循环中释放它,但它释放了所有变量。这是我的尝试

for(int i=0;i<vl.size();i++) {
    gcppoint_ *a = qobject_cast<gcppoint_ *>(vl.at(i).value<QObject *>());

      char* srcX =strdup(const_cast<char*>(QString::number(a->row()).toStdString().c_str()));
      char* srcY = strdup(const_cast<char*>(QString::number(a->column()).toStdString().c_str()));
      char* dstX =strdup( const_cast<char*>(QString::number(a->lon()).toStdString().c_str()));
      char* dstY = strdup(const_cast<char*>(QString::number(a->lat()).toStdString().c_str()));
      qDebug() <<srcX<<" " <<srcY << " " <<dstX<< " " <<dstY;
     argvarry[i]=srcX;

      if(srcX)
          free(srcX) //does not work it frees argvarry[i] too

}
//...do some thing with argvarry
free????

我注意到所有argvarry[i] 也将被释放,所以我不能再使用它们了。好吧,我怎样才能让它工作,我太空闲了strdup,但我无法处理它。我怎样才能改变这个循环,当我使用argvarry我释放strup结果?


如果我把上面的代码改成类似的会怎样

char * srcX;
char * srcY;
char * dstX;
char * dstY;

for (int i = 0;i < gcps.size(); i++) {
    argvarry[argc] = "-gcp";argc++;//gcp_values
    gcppoint_ *a = qobject_cast<gcppoint_ *>(gcps.at(i).value<QObject *>());

      srcX= strdup( const_cast<char*>(QString::number(a->row()).toStdString().c_str())); 
      srcY = strdup(const_cast<char*>(QString::number(a->column()).toStdString().c_str())); 
      dstX = strdup(const_cast<char*>(QString::number(a->lon()).toStdString().c_str())); 
      dstY =strdup( const_cast<char*>(QString::number(a->lat()).toStdString().c_str()));
      argvarry[argc]=srcX;argc++;
      argvarry[argc]=srcY;argc++;
      argvarry[argc]=dstX;argc++;
      argvarry[argc]=dstY;argc++;
      }
          ///do things

         //free(srcX);free(srcY)

【问题讨论】:

  • 由于您使用的是 C++,而不是 C,因此您不需要像 strdup()free() 这样的函数。只需使用std::vector&lt;std::string&gt;(或QString 的向量,或QStringList,或Qt 推荐的任何东西)。
  • @KarstenKoop 嗨,但我需要填写 char * argvarry[(gcps.size() * 5) + 8];而且我最需要将 Qstring 和 std:string 转换为 char * 并且我将再次面临让它们免费和这样的问题
  • @MajidHojati -- 我相信如果没有strdup 调用,这可以做得更好。

标签: c++ strdup


【解决方案1】:
  char* srcX = const_cast<char*>(QString::number(a->row()).toStdString().c_str());

上面的问题是 toStdString() 返回一个临时的 QString 对象,并且由于 QString 对象是临时的,它在行尾被销毁。这意味着当您稍后尝试使用 srcX 时,它是一个悬空指针。

如果您绝对需要char * 指针数组,则需要确保它们指向的数据在数组的整个生命周期内保持有效。 strdup() 是一种方法,但正如您所指出的,它有其自身的困难,特别是您需要在完成字符串后手动调用 free() ,否则您会泄漏内存.

另一种方法是先建立一个 std::string 对象列表,如下所示:

// Build up a list of std::strings
std::vector<std::string> strsList;
strsList.push_back("-of");
strsList.push_back("GTiff");
strsList.push_back("-a_nodata");
strsList.push_back("'0 0 0'");
strsList.push_back("-a_srs");
strsList.push_back("a_srs");
for (int i = 0;i < gcps.size(); i++) {
   strsList.push_back("-gcp");
   strsList.push_back(QString::number(a->row()).toStdString());
   strsList.push_back(QString::number(a->column()).toStdString());
   strsList.push_back(QString::number(a->lon()).toStdString());
   strsList.push_back(QString::number(a->lat()).toStdString());
}

// Now create an array of pointers to the data in those std::strings
// This array will remain valid for as long as strsList remains valid and unmodified
char * argvarray[strsList.size()];
for (size_t i=0; i<strsList.size(); i++) argvarray[i] = const_cast<char *>(strsList[i].c_str());

【讨论】:

  • 您好,感谢您的帮助,关于第一部分我在每行末尾使用 argc++。所以它增加了。 argvarry 必须是 char * 并且我无法转换 std::string 和 Qstring info char * 例如某些函数将它们转换为 const char * 并且它不起作用,请指导我在使用后释放 strup 输出他们?
  • 你能检查我的问题吗?我添加了一个新方法,你认为它安全吗?
  • 你的新方法看起来很安全(只要我们愿意假设 strdup() 永远不会返回 NULL),但如果你不小心释放它可能会泄漏内存( ) 你 strdup()'d 的所有字符串。 (请注意,您还需要小心在您没有 strdup() 的任何字符串上调用 free(),并且因为您正在存储指向 strdup() 的指针,并且静态分配的字符串都放在同一个 char 数组中,代码可能非常脆弱且容易出错)
  • 感谢您的帮助,我正在尝试您提出的方法我会分享结果,谢谢
  • "也就是说srcX是一个悬空指针":srcXstrdup的返回值,不是QString的临时返回值。
猜你喜欢
  • 2021-09-24
  • 2014-09-20
  • 2011-06-25
  • 1970-01-01
  • 1970-01-01
  • 2011-11-14
  • 2020-09-24
  • 2019-10-11
  • 2015-09-15
相关资源
最近更新 更多