背景
使用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 格式也不行
结论
所以怀疑硬的硬识别有问题,还有其他几个字识别有问题。目前暂时实现不了中文读取,所以先改成英文了。