《Qt入门 (九)--QtJSON读取中文出现的问题》

背景

使用Qt自带的api读取JSON 文件,由于为了方便一些文件的配置,让普通人也能配置文件,所以想让JSON的文件使用中文的Key来显示,但是暂时没实现该功能,记录一下遇到的几个问题。

源码

啥都不说先贴源码
mdljsconfig.h 的头文件

#ifndef MDLJSCONFIG_H
#define MDLJSCONFIG_H

#include <QVariantMap>
#include <QFile>
#include <QJsonDocument>
#include <QJsonObject>
#include <QJsonParseError>
#include <QJsonArray>
#include <QJsonValue>

#include <QDebug>

class JsConfig
{
public:
    JsConfig(const QString &fileName);
    ~JsConfig();

    bool open(const QString &fileName);
    void sync();

    void write(const QString &key, const QVariant& value);
    QVariant readVariant(const QString &key, const QVariant &value);
    QString  readString(const QString &key, const QString &value = "");
    bool readBool(const QString &key, bool value = false);
    int readInt(const QString &key, int value = 0);
    QJsonObject getJsonObject();
private:
    QString m_fileName;
    QVariantMap m_cache;
    QJsonObject m_valConfigObject;
};

#endif // SINGLETON

mdljsconfig.cpp头文件

#include "mdljsconfig.h"

# pragma execution_character_set("utf-8")

JsConfig::JsConfig(const QString &fileName)
    : m_fileName(fileName)
{
    open(fileName);
}


JsConfig::~JsConfig()
{
    sync();
}

bool JsConfig::open(const QString &fileName)
{
    QFile file(fileName);

    if (!file.open(QIODevice::ReadOnly)){
        qDebug()<<"open erro";
        return false;
    }
    QByteArray allData = file.readAll();
    file.close();

    QJsonParseError jsonError;
    QJsonDocument jsonDoc = QJsonDocument::fromJson(allData, &jsonError);

    if (jsonError.error != QJsonParseError::NoError){
        qDebug()<<"anilyz erro"<<jsonError.error;
        return false;
    }

    m_valConfigObject = jsonDoc.object();
    //QList<QVariant> MonitorList = jsonDoc.toVariant().toMap()["mainInfo"].toList();

#if 0
    QList<QVariant> slist = m_valConfigObject.toVariantMap()["Value"].toList();
    for (auto &item: slist)
    {
        QVariantMap sMap = item.toMap();
        qDebug() << sMap["功能"];
        qDebug() << sMap["硬件类型"];
        qDebug() << sMap["name"];

    }
#if 0
    if(m_valConfigObject.contains(QStringLiteral("Value"))){
         QJsonValue arrayValue = m_valConfigObject.value(QStringLiteral("Value"));

         if(arrayValue.isArray()){
            QJsonArray array = arrayValue.toArray();
            for(int i=0;i<array.size();i++){
                QJsonValue iconArray = array.at(i);
                QJsonObject icon = iconArray.toObject();

                QString id = icon["id"].toString();
                QString gongneng = icon["硬件类型"].toString();
                QString name = icon["name"].toString();
                qDebug()<< array.at(i);
                qDebug() << id<< name;
            }
         }else{
             qDebug() <<"no arry";
         }
    }else{
        qDebug() <<"no value";
    }
  #endif
#else
    m_cache = m_valConfigObject.toVariantMap();
    QStringList keys = m_valConfigObject.keys();

//    for(int i = 0; i < m_cache.size(); i++){

//        QString icon = m_cache.value("1").toString();
//        qDebug() <<"发送类型"<<icon;
//    }

    qDebug() <<"m_valConfigObject"<< m_valConfigObject<<"\n";
    qDebug() <<"0:"<< m_valConfigObject.value("0").toObject()<<"\n";
    qDebug() <<"1:"<< m_valConfigObject.value("1").toObject()<<"\n";

    for(int i = 0; i < keys.size(); i++){

        //QJsonObject icon =  m_valConfigObject.value(keys.at(i)).toObject();
         QJsonObject icon =    m_valConfigObject.value(keys.at(i)).toObject();
         qDebug() <<"取值前"<< icon;
         qDebug() << "取值前是否存在注释:"<< icon.contains("注释");
         qDebug() << "取值前是否存在发送类型:"<< icon.contains("件");
         qDebug() << "取值前是否存在硬件类型:"<< icon.contains("发送");
#if 0
        int id   = icon.value("id").toInt();
        QString name = icon.value("name").toString();

        QString function  = icon["功能"].toString();
        QString hardtype  = icon["硬件类型"].toString();
        QString sendtype  = icon["发送类型"].toString();
        int maxval  = icon["最大值"].toInt();
        int minval  = icon["最小值"].toInt();
        QString psss = icon["注释"].toString();

        qDebug() <<"取值后"<< icon;
        qDebug() <<"id="      <<id;
        qDebug() <<"name="    <<name;
        qDebug() <<"功能="     <<function;
        qDebug() <<"硬件类型=" <<hardtype;
        qDebug() <<"发送类型=" <<sendtype;
        qDebug() <<"最大值="  <<maxval;
        qDebug() <<"最小值="  <<minval;
        qDebug() <<"注释="    <<psss;
#else
         int id   = icon.value("id").toInt();
         QString name = icon.value("name").toString();

         QString function  = icon.value("功能").toString();

         QString hardtype  = icon.value("件").toString();
         QString sendtype  = icon.value("发送").toString();
         int maxval  = icon.value("最大值").toInt();
         int minval  = icon.value("最小值").toInt();
         QString psss = icon.value("注释").toString();

         qDebug() <<"取值后"<< icon;
         qDebug() << "是否存在发送类型:"<< icon.contains("发送类型");
         qDebug() <<"id="      <<id;
         qDebug() <<"name="    <<name;
         qDebug() <<"功能="     <<function;
         qDebug() <<"硬件类型=" <<hardtype;
         qDebug() <<"发送类型=" <<sendtype;
         qDebug() <<"最大值="  <<maxval;
         qDebug() <<"最小值="  <<minval;
         qDebug() <<"注释="    <<psss;


#endif
    }
#endif
    return true;
}

void JsConfig::sync()
{
    QJsonObject root = QJsonObject::fromVariantMap(m_cache);
    QJsonDocument jsonDoc(root);
    QByteArray data = jsonDoc.toJson(QJsonDocument::Compact);
    QFile file(m_fileName);

    if (file.open(QIODevice::WriteOnly)){
        file.write(data);
        file.close();
    }
}

void JsConfig::write(const QString &key, const QVariant &value)
{
    m_cache.insert(key, value);
}

QVariant JsConfig::readVariant(const QString &key, const QVariant &value)
{
    if (m_cache.contains(key)){
        return m_cache.value(key).toMap();
    }

    return value;
}

QString JsConfig::readString(const QString &key, const QString &value)
{
    if (m_cache.contains(key)){
        return m_cache.value(key).toString();
    }

    return value;
}

bool JsConfig::readBool(const QString &key, bool value)
{
    if (m_cache.contains(key)){
        return m_cache.value(key).toBool();
    }

    return value;
}



int JsConfig::readInt(const QString &key, int value){
    if (m_cache.contains(key)){
        return m_cache.value(key).toInt();
    }

    return value;
}

QJsonObject JsConfig:: getJsonObject()
{
    return  m_valConfigObject;
}
//int JsConfig::getKeynum(int value){

//    if (m_cache.contains(key)){
//        return  m_cache.count();;
//    }

//    return value;
//}

json 文件

{
    "0":{
            "id": 0,
            "功能": "nomal",
            "硬件类型": "key",
            "发送类型": "RREF",
            "name": "sim/cockpit/autopilot/flight_director_roll",
            "取反": 1,
            "最大值": 1,
            "最小值": 1,
            "中间值": 1,
            "注释": "roll"
        },
	"1":{
            "id": 1,
            "功能": "nomal",
            "硬件类型": "key",
            "发送类型": "RREF",
            "name": "sim/cockpit/autopilot/flight_director_pitch",
            "取反": 1,
            "最大值": 1,
            "最小值": 1,
            "中间值": 1,
            "注释": "pitch"
        },
	"2":{
            "id": 2,
            "功能": "nomal",
            "硬件类型": "key",
            "发送类型": "RREF",
            "name": "sim/cockpit/autopilot/heading",
            "取反": 1,
            "最大值": 1,
            "最小值": 1,
            "中间值": 1,
            "注释": "yaw"
        }
}

问题一(中文转义)

因为key是中文在解析的时候 需要增加中文的转义
尝试了几种转义后还是# pragma execution_character_set(“utf-8”)最为方便

# pragma execution_character_set("utf-8")
QLatin1String()        //转义也可以但是每个都要添加比较麻烦
QString::fromLocal8Bit//转义也可以但是每个都要添加比较麻烦

问题二(读取方式)

使用icon[“注释”].toString();中括号取值的方式,会导致取不到值时
Qt应该是为了防止取到其他错误的值,而给你写了个值为null的key

在这里插入图片描述
在这里插入图片描述
所以导致取值完后会多了一些null值(部分中文可以取值,“硬件类型”则取不出来。暂时不懂为啥)
所以后续改用 value()的方式取值(虽然也取不出),但是不会额外增加值了
在这里插入图片描述
在这里插入图片描述

测试

后续又在中文那边做了几个测试。
1、4个字“硬件类型”改为“硬件”(读不出)
2、因为功能可以正常读出所以把“硬件类型”改为“功效”,可以正常读出
3、“硬件类型”改为“件” 可以正常读出。
4、json 文件使用vscode 转码成utf8 格式也不行

结论

所以怀疑硬的硬识别有问题,还有其他几个字识别有问题。目前暂时实现不了中文读取,所以先改成英文了。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值