UCHAR to Qstring in QT C++



我在qt c 中具有功能:

UCHAR GetResult(HANDLE Handle, UCHAR *Address, UCHAR *Status, int *Value)
{
    UCHAR RxBuffer[9], Checksum;
    DWORD Errors, BytesRead;
    COMSTAT ComStat;
    int i;
    //Check if enough bytes can be read
    ClearCommError(Handle, &Errors, &ComStat);
    if(ComStat.cbInQue>8)
    {
        //Receive
        ReadFile(Handle, RxBuffer, 9, &BytesRead, NULL);
        Checksum=0;
        for(i=0; i<8; i++)
            Checksum+=RxBuffer[i];
        if(Checksum!=RxBuffer[8]) return TMCL_RESULT_CHECKSUM_ERROR;
        *Address=RxBuffer[0];
        *Status=RxBuffer[2];
        *Value=(RxBuffer[4] << 24) | (RxBuffer[5] << 16) | (RxBuffer[6] << 8) | RxBuffer[7];
    } else return TMCL_RESULT_NOT_READY;
    return TMCL_RESULT_OK;
}

它返回UCHAR的值:

  • TMCL_RESULT_CHECKSUM_ERROR
  • TMCL_RESULT_NOT_READY
  • TMCL_RESULT_OK

基于输入参数。我现在想在QLabel中显示返回,但是问题当然是UCHAR,而不是字符串变量。

如何将这些值转换为QString,以便我可以输出它们。

我想到说三个语句,但认为有更好的方法。例如:

if (GetResult(RS232Handle, &Address, &Status, &Value)==TMCL_RESULT_OK) {
    printf("TMCL_RESULT_OK");
}

首先, GetResult既不是qt也不是c 。它是C代码。它可能会在1993年左右使用任何32位Windows C编译器进行编译。

如果您想现代化它,则可以使用QSerialPort

// https://github.com/KubaO/stackoverflown/tree/master/questions/serial-controller-40754585
#include <QtWidgets>
#include <QtSerialPort>
class MyController : public QObject {
  Q_OBJECT
public:
  enum IoStatus { Closed, NotReady, Ok, ChecksumError, PortError, Invalid = -1 };
private:
  QSerialPort m_port{this};
  QDataStream m_str{&m_port};
  IoStatus m_ioStatus = Invalid;
  uint8_t m_address, m_status;
  uint32_t m_value;
  bool check(const QByteArray &packet) {
    char checksum = 0;
    for (int i = 0; i < packet.size()-1; ++i)
      checksum += packet [i];
    return checksum == packet[packet.size()-1];
  }
  void onError(QSerialPort::SerialPortError err) {
    if (err != QSerialPort::NoError)
      setIoStatus(PortError);
  }
  void onRxData() {
    if (m_port.bytesAvailable() < 9) return;
    if (m_port.error() != QSerialPort::NoError)
      return;
    if (! check(m_port.peek(9)))
      return setIoStatus(ChecksumError);
    uint8_t dummy;
    m_str >> m_address >> dummy >> m_status >> dummy >> m_value;
    setIoStatus(Ok);
  }
  void setIoStatus(IoStatus ioStatus) {
    if (m_ioStatus == ioStatus) return;
    m_ioStatus = ioStatus;
    emit ioStatusChanged(m_ioStatus);
  }
  static QString text(IoStatus ioStatus) {
    switch (ioStatus) {
    case NotReady: return "Not Ready";
    case Ok: return "Ok";
    case ChecksumError: return "Checksum Error";
    case PortError: return "Serial Port Error";
    default: return "Unknown Status";
    }
  }
public:
  explicit MyController(QObject *parent = nullptr) : QObject(parent) {
    connect(&m_port, &QIODevice::readyRead, this, &MyController::onRxData);
    connect(&m_port, static_cast<void(QSerialPort::*)(QSerialPort::SerialPortError)>(&QSerialPort::error), this, &MyController::onError);
    m_str.setByteOrder(QDataStream::BigEndian);
    QTimer::singleShot(0, this, [this]{ setIoStatus(Closed); });
  }
  bool open() {
    auto rc = m_port.open(QIODevice::ReadWrite);
    if (rc) setIoStatus(NotReady);
    return rc;
  }
  IoStatus ioStatus() const { return m_ioStatus; }
  QString ioStatusText() const { return text(m_ioStatus); }
  Q_SIGNAL void ioStatusChanged(IoStatus);
  QSerialPort *port() { return &m_port; }
};

它可以大约使用如下 - 这只是您如何与控制器接口的一个示例:

int main(int argc, char **argv) {
  QApplication app{argc, argv};
  MyController ctl;
  QWidget w;
  QFormLayout layout{&w};
  QLineEdit port;
  QLabel status;
  QPushButton open{"Open"};
  layout.addRow("Port", &port);
  layout.addRow(&status);
  layout.addRow(&open);
  QObject::connect(&open, &QPushButton::clicked, [&]{
    ctl.port()->setPortName(port.text());
    ctl.open();
  });
  QObject::connect(&ctl, &MyController::ioStatusChanged, [&]{
    status.setText(ctl.ioStatusText());
  });
  w.show();
  return app.exec();
}
#include "main.moc"

如果您坚持使用旧的C代码,则可以通过在结构中返回结果来进行现代化:

struct Result {
  uint8_t address;
  uint8_t status;
  uint32_t value;
  enum Status { Ok, NotReady, ChecksumError, ReadError };
  Status ioStatus = Ok;
  QString ioStatusText() const {
    switch (ioStatus) {
    case NotReady: return "Not Ready";
    case Ok: return "Ok";
    case ChecksumError: return "Checksum Error";
    case ReadError: return "Read Error";
    default: return "Unknown Status";
    }
  }
};
Result getResult(HANDLE handle)
{
  Result result;
  DWORD errors, bytesRead;
  COMSTAT comStat;
  ClearCommError(handle, &errors, &comStat);
  if (comStat.cbInQue < 9) {
    result.ioStatus = Result::NotReady;
    return result;
  }
  uint8_t buffer[9];
  ReadFile(handle, buffer, 9, &bytesRead, NULL);
  if (bytesRead < 9) {
    result.ioStatus = Result::ReadError;
    return result;
  }
  uint8_t checksum=0;
  for (int i=0; i<8; i++)
    checksum += buffer[i];
  if (checksum != buffer[8]) {
    result.ioStatus = Result::ChecksumError;
    return result;
  }
  result.address = buffer[0];
  result.status = buffer[2];
  result.value = (buffer[4] << 24) | (buffer[5] << 16) | (buffer[6] << 8) | buffer[7];
  return result;
}

然后您可以做:

auto result = getResult(handle);
printf("%sn", result.ioStatusText().toLocal8Bit().constData());

function getResult()返回的值只是一个UCHAR值,而不是指向Uchars容器的指针。

所以,如果我理解您的问题,我认为最快的方法(根据您提供的信息)是在您的QT项目中执行此操作:

QString text_to_display;
uchar result = GetResult(RS232Handle, &Address, &Status, &Value);
switch(result)
{
    case TMCL_RESULT_OK:
         text_to_display = "TMC_RESULT_OK";
         break;
    case TMCL_RESULT_NOT_READY:
         text_to_display = "TMCL_RESULT_NOT_READY";
         break;
    case TMCL_RESULT_CHECKSUM_ERROR:
         text_to_display = "TMCL_RESULT_CHECKSUM_ERROR";
         break;
    default:
         break;
}
QLabel *label_to_show_result = new QLabel(text_to_display, this);

这不会改变您的 geresult 函数,只需取得结果并根据Qlabel显示文本。

当然,您可以使用其他方式,例如 qhash 映射结果文本的夫妻,避免开关案例结构。

最新更新