【问题标题】:Cannot specify explicit initializer for arrays in Windows Qt 5.7 (C++)无法在 Windows Qt 5.7 (C++) 中为数组指定显式初始化程序
【发布时间】:2016-09-28 20:42:56
【问题描述】:

跟进一个很棒的answer,我正在尝试解决一个明显常见的错误**C2536 : cannot specify explicit initializer for arrays** for line in line: QLineEdit * edits[3] = {&lineEditName, &lineEditGender, &lineEditRegion};

我已阅读其他解决方案(例如12),但似乎不适用于我的情况。我在 Windows 中使用 Qt Creator 4.1.0 和 Qt 5.7.0,MSVC 2013 和 C++。欢迎提出任何建议。

额外说明一下,为了更好地理解 Qt 与 C++,有人可以指出一个不同的方式 -而不是使用数组- 来保存新的解析json结果?

您可以在下面看到该答案的完整代码(有问题的行在mainwindow.h):

main.cpp

// https://github.com/KubaO/stackoverflown/tree/master/questions/into-mainwin-39643510
#include "mainwindow.h"
#include "controller.h"

int main(int argc, char *argv[])
{
   QApplication app{argc, argv};
   MainWindow ui;
   Controller ctl;
   QTimer timer;
   timer.start(5*1000);
   QObject::connect(&timer, &QTimer::timeout, &ctl, &Controller::get);

   QObject::connect(&ctl, &Controller::busy, &ui, [&]{ ui.setState(MainWindow::Loading); });
   QObject::connect(&ui, &MainWindow::request, &ctl, &Controller::get);
   QObject::connect(&ctl, &Controller::error, &ui, [&]{ ui.setState(MainWindow::Error); });
   QObject::connect(&ctl, &Controller::values, &ui, &MainWindow::setFields);
   ui.show();
   return app.exec();

控制器.h

#ifndef CONTROLLER_H
#define CONTROLLER_H

#include <QtNetwork>

class Controller : public QObject {
   Q_OBJECT
   QNetworkAccessManager manager{this};
   QNetworkRequest request;
   Q_SLOT void onReply(QNetworkReply *);
public:
   explicit Controller(QObject * parent = nullptr);
   Q_SLOT void get();
   Q_SIGNAL void busy();
   Q_SIGNAL void error(const QString &);
   Q_SIGNAL void values(const QString & name, const QString & gender, const QString & region);
};

#endif // CONTROLLER_H

控制器.cpp

#include "controller.h"

Controller::Controller(QObject *parent) : QObject(parent)
{
   QUrlQuery query;
   query.addQueryItem("amount", "1");
   query.addQueryItem("region", "United States");
   QUrl url("http://uinames.com/api/");
   url.setQuery(query);
   request = QNetworkRequest(url);
   connect(&manager, &QNetworkAccessManager::finished, this, &Controller::onReply);
}

void Controller::onReply(QNetworkReply * reply) {
   if (reply->error() != QNetworkReply::NoError) {
      emit error(reply->errorString());
      manager.clearAccessCache();
   } else {
      //parse the reply JSON and display result in the UI
      auto jsonObject = QJsonDocument::fromJson(reply->readAll()).object();
      auto fullName = jsonObject["name"].toString();
      fullName.append(" ");
      fullName.append(jsonObject["surname"].toString());
      emit values(fullName, jsonObject["gender"].toString(), jsonObject["region"].toString());
   }
   reply->deleteLater();
}

void Controller::get() {
   emit busy();
   manager.get(request);
}

主窗口.h

#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QtWidgets>

class MainWindow : public QWidget {
  Q_OBJECT
  QFormLayout layout{this};
  QLineEdit lineEditName;
  QLineEdit lineEditGender;
  QLineEdit lineEditRegion;
  QPushButton button{"Get Name"};
  QLineEdit * edits[3] = {&lineEditName, &lineEditGender, &lineEditRegion};
public:
  enum State { Normal, Loading, Error };
  explicit MainWindow(QWidget * parent = nullptr);
  Q_SLOT void setFields(const QString & name, const QString & gender, const QString & region);
  Q_SLOT void setState(State);
  Q_SIGNAL void request();
};

#endif // MAINWINDOW_H

主窗口.cpp

#include "mainwindow.h"

MainWindow::MainWindow(QWidget *parent) : QWidget(parent)
{
   for(auto edit : edits) edit->setReadOnly(true);
   layout.addRow("Name:", &lineEditName);
   layout.addRow("Gender:", &lineEditGender);
   layout.addRow("Region:", &lineEditRegion);
   layout.addRow(&button);
   connect(&button, &QPushButton::clicked, this, &MainWindow::request);
}

void MainWindow::setFields(const QString & name, const QString & gender, const QString & region) {
   setState(Normal);
   lineEditName.setText(name);
   lineEditGender.setText(gender);
   lineEditRegion.setText(region);
}

void MainWindow::setState(MainWindow::State state) {
   if (state == Normal) {
      for (auto edit : edits) edit->setEnabled(true);
      button.setEnabled(true);
   }
   else if (state == Loading) {
      for (auto edit : edits) edit->setEnabled(false);
      button.setEnabled(false);
   }
   else if (state == Error) {
      for (auto edit : edits) edit->setText("Error...");
      button.setEnabled(true);
   }
}

【问题讨论】:

  • 如果:QLineEdit * 编辑[] {&lineEditName, &lineEditGender, &lineEditRegion};
  • @AlexanderVX No 也没有解决它,它与an array of unknown size cannot be used in a range-based for statement 给出相同的错误。有任何其他建议或提供edits 数组的替代方案吗?

标签: c++ arrays qt qt-creator


【解决方案1】:

问题在于 MSVC2013 does not support 所有 C++11 功能。这里的主要问题是您不能为成员数组指定类内初始化。

在我看来,最接近的解决方法是使用std::array 而不是edits 的C 数组,因为MSVC2013 似乎支持使用aggregate initialization 初始化std::array,您可以替换您的:

QLineEdit * edits[3] = {&lineEditName, &lineEditGender, &lineEditRegion};

使用std::array 并像这样聚合初始化:

std::array<QLineEdit*,3> edits{{&lineEditName, &lineEditGender, &lineEditRegion}};

请注意,您必须将#include &lt;array&gt; 添加到您的mainwindow.h 才能使用std::array

额外说明,为了更好地理解 Qt C ++,有人可以指出不同的方式-而不是使用数组- 用于保存新解析的 json 结果?

您没有使用 arrays 来保存新的解析 json 结果,这正是您使用 QJsonObject 的方式。

如果您正在寻找一种在其自己的类中进一步分离 JSON 解析的方法。您可能希望拥有一些封装名称数据(fullNamegenderregion)的类(例如NameData),并提供一种由 JSON 数据的QByteArray 构造的方法。

这样Controller 类不知道任何关于JSON 的信息,它只是使用NameData 类来解析它从网络收到的回复。

你可以这样实现它:

名称数据.h

#ifndef NAMEDATA_H
#define NAMEDATA_H

#include <QtCore>

class NameData
{
public:
    QString fullName;
    QString gender;
    QString region;

    static NameData fromReplyByteArray(QByteArray byteArray);
};

#endif // NAMEDATA_H

名称数据.cpp

#include "namedata.h"

NameData NameData::fromReplyByteArray(QByteArray byteArray){
    QJsonObject jsonObject = QJsonDocument::fromJson(byteArray).object();
    NameData result;
    result.fullName = jsonObject["name"].toString();
    result.fullName.append(" ");
    result.fullName.append(jsonObject["surname"].toString());
    result.gender= jsonObject["gender"].toString();
    result.region= jsonObject["region"].toString();
    return result;
}

现在,Controller 类中的 onReply 函数应该是:

void Controller::onReply(QNetworkReply * reply) {
    if (reply->error() != QNetworkReply::NoError) {
        emit error(reply->errorString());
        manager.clearAccessCache();
    } else {
        //parse the reply JSON and display result in the UI (using the separate class)
        NameData nameData= NameData::fromReplyByteArray(reply->readAll());
        emit values(nameData.fullName, nameData.gender, nameData.region);
    }
    reply->deleteLater();
}

【讨论】:

    猜你喜欢
    • 2015-03-09
    • 1970-01-01
    • 1970-01-01
    • 2023-03-21
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多