从QString的QMap创建char数组的问题

时间:2018-05-14 01:51:44

标签: c++ arrays qt char

我使用libxslt C库,需要传递参数const char *。我将库包装在Qt C ++类中,因此存储在C ++类中的参数存储为QMap<QString, QString>

我的第一次尝试就是:

const char *params[32];
int index = 0;
if (m_params.size() > 0) {
    QMapIterator<QString, QString> it(m_params);
    while (it.hasNext()) {
        it.next();

        params[index++] = it.key().toLocal8Bit().data();

        params[index++] = it.value().toLocal8Bit().data();
    }
}

params[index++] = nullptr;

qDebug() << params[0] << params[1]; // 0 0

但我意识到这不起作用,因为QByteArray的{​​{1}}几乎在我使用它之后就会超出范围。

我尝试过使用strcpy - 但是具有相同的范围问题:

toLocal8bit

所以现在我有一个具有相同值的params列表。

当我手动设置所有值时,我得到了预期的结果:

m_params.insert("some-key", "some-value", "another-key", "another-value");

if (m_params.size() > 0) {
    QMapIterator<QString, QString> it(m_params);
    while (it.hasNext()) {
        it.next();

        char buffer[32];

        strcpy(buffer, it.key().toLocal8Bit().data());
        params[index++] = buffer;

        strcpy(buffer, it.value().toLocal8Bit().data());
        params[index++] = buffer;
    }
}

params[index++] = nullptr;

qDebug() << params[0] << params[1]; // another-value another-value

1 个答案:

答案 0 :(得分:0)

这很容易 - 您需要确保参数的缓冲区持续足够长的时间。不要使用固定大小的数组 - 你要为缓冲区溢出做好准备。

class Params {
  QByteArray buf;
  QVector<const char *> params;
public:
  Params() = default;
  template <class T> explicit Params(const T& map) {
    QVector<int> indices;
    indices.reserve(map.size());
    params.reserve(map.size()+1);
    for (auto it = map.begin(); it != map.end(); ++it) {
      indices.push_back(buf.size());
      buf.append(it.key().toLocal8Bit());
      buf.append('\0');
      indices.push_back(buf.size());
      buf.append(it.value().toLocal8Bit());
      buf.append('\0');
    }
    for (int index : qAsConst(indices))
      params.push_back(buf.constData() + index);
    params.push_back(nullptr);
  }
  operator const char **() const { return const_cast<const char**>(params.data()); }
  operator const char *const*() const { return params.data(); }
  operator QVector<const char*>() const { return params; }
};

void MyClass::method() const {
  Params params{m_params};
  ...
  res = xsltApplyStylesheet(cur, doc, params);
  ...
}