【QT常用技术讲解】QSettings把中文输入到配置文件

前言

        在 QT 中,使用 QSettings 时,默认是将字符串以 Unicode 格式存储,而不是以 UTF-8 编码直接写入配置文件。因为涉及到配置文件,有些时候,配置信息由界面端进行写操作,而后台服务进程进行读取并处理,碰巧便于维护的要求,某些配置项需要中文时,就需要设置为UTF-8格式存储。

QSettings特性

1、默认以 Unicode 格式存储;

2、受特殊字符影响,比如①内容包含逗号','时,读出的配置内容为空;②内容包含分号';'时,读出的配置内容被截断。如果想要读取,配置内容需要加上引号"";

3、配置项不能为中文,比如QString confstr = settings.value("哈哈").toString();取不到值。

如果配置文件配置项/配置内容,比较复杂,不建议使用QSetting,使用原生的文件读取操作:逐行读取->=号分割->匹配关键字->返回内容项(文章末尾提供源码)。

功能讲解

QString设置UTF-8编码

#include <QSettings>
#include <QDebug>

void MainWindow::saveconfmsg(){
    QSettings settings("config.ini", QSettings::IniFormat);//打开文件
    
    QString savestr="测试";
    // 保存中文字符
    settings.setValue("ChineseText", savestr);
}

void MainWindow::getconfmsg(){
    QSettings settings("config.ini", QSettings::IniFormat);//打开文件

    // 读取中文字符
    QString confText = settings.value("ChineseText").toString();
    qDebug() << confText ;

}

通常,如果配置文件的配置项只是QT界面端进行调用的话,以上的保存和读取函数已经够用了,但此时config.ini配置文件中保存的内容是Unicode编码,如下图所示:

而且在测试过程中还发现了会出现多加上了双引号的情况,此时只需要设置为UTF-8方式,即可解决此类问题,代码如下:

#include <QSettings>
#include <QDebug>

void MainWindow::saveconfmsg(){
    QSettings settings("config.ini", QSettings::IniFormat);//打开文件
    
    settings.setIniCodec("UTF-8");//-------设置为UTF-8方式

    QString savestr="测试";
    // 保存中文字符
    settings.setValue("ChineseText", savestr);
}

void MainWindow::getconfmsg(){
    QSettings settings("config.ini", QSettings::IniFormat);//打开文件

    settings.setIniCodec("UTF-8");//-------设置为UTF-8方式

    // 读取中文字符
    QString confText = settings.value("ChineseText").toString();
    qDebug() << confText ;

}

另外,本人测试了QTextCodec的方式无效

QTextCodec *codec = QTextCodec::codecForName("UTF-8");

 Unicode转utf-8

如果界面端已经把中文以Unicode编码方式写入到配置文件,后台服务程序又是普通的C++程序,这时就需要进行Unicode转utf-8

#include <vector>
#include <iomanip>
#include <chrono>
#include <string>
#include <dlfcn.h>
#include <sstream>
#include <fstream>
// 将 Unicode 码点转换为 UTF-8 字符串
std::string unicodeToUtf8(unsigned int codepoint) {
    std::string utf8;
    if (codepoint <= 0x7F) {
        utf8.push_back(static_cast<char>(codepoint));
    } else if (codepoint <= 0x7FF) {
        utf8.push_back(static_cast<char>((codepoint >> 6) | 0xC0));
        utf8.push_back(static_cast<char>((codepoint & 0x3F) | 0x80));
    } else if (codepoint <= 0xFFFF) {
        utf8.push_back(static_cast<char>((codepoint >> 12) | 0xE0));
        utf8.push_back(static_cast<char>(((codepoint >> 6) & 0x3F) | 0x80));
        utf8.push_back(static_cast<char>((codepoint & 0x3F) | 0x80));
    } else {
        utf8.push_back(static_cast<char>((codepoint >> 18) | 0xF0));
        utf8.push_back(static_cast<char>(((codepoint >> 12) & 0x3F) | 0x80));
        utf8.push_back(static_cast<char>(((codepoint >> 6) & 0x3F) | 0x80));
        utf8.push_back(static_cast<char>((codepoint & 0x3F) | 0x80));
    }
    return utf8;
}

// 解析 Unicode 字符串并转换为 UTF-8
std::string convertUnicodeStringToUtf8(const std::string& input) {
    std::string result;
    bool bflag=false;
    char flag1='\\';
    char flag2='x';
    size_t strlen=input.size();
    for(size_t i=0;i<strlen;i++){
        char ch=input[i];
        if(ch==flag1) {
            bflag=true;
            continue;
            //printf("i=%ld\n",i);
        }
        if(bflag==true && ch==flag2){
            if(((i+2)<strlen && input[i+3]==flag1)||(i+3)==strlen){//数字类型:\x31\x32\x33
                std::string hexStr = input.substr(i + 1, 2);
                unsigned int codepoint = std::stoul(hexStr, nullptr, 16);  // 转换为整数
                std::cout << hexStr << "=>" << unicodeToUtf8(codepoint) << std::endl;
                i+=2;
                result+=unicodeToUtf8(codepoint);
            }else if((i+5)<=strlen){
                std::string hexStr = input.substr(i + 1, 4);
                unsigned int codepoint = std::stoul(hexStr, nullptr, 16);  // 转换为整数
                //std::cout << hexStr << "=>" << unicodeToUtf8(codepoint) << std::endl;
                i+=4;
                result+=unicodeToUtf8(codepoint);
            }
        }else{
            bflag=false;
            //printf("i=%ld ch=%c\n",i,ch);
            result+=ch;
        }
    }
    //std::cout << "=>" << result << std::endl;
    return result;
}


int main(){
   Getbaseinfo(CONFIGPATH);//需要自己实现
   std::string utf8_hanzistr = convertUnicodeStringToUtf8(conf_hanzistr);//====转码

}

注:以上的convertUnicodeStringToUtf8()未去掉双引号。

QFile读取配置文件方式

#include <QFile>
#include <QTextStream>
#include <QDebug>
QString getconfmsgbyflag(const QString& flag){
    QString result;
    QFile file(CONFPATH); // 创建文件对象
    if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { // 打开文件
        qDebug() << "无法打开文件!";
        return result;
    }
    QTextStream in(&file); // 创建文本流
    in.setCodec("UTF-8");
    while (!in.atEnd()) { // 逐行读取
        QString line = in.readLine(); // 读取一行
        qDebug() << line; // 输出每一行
        QStringList conflist = line.split("=");
        if(conflist.size()!=2) continue;
        QString conf_flag=conflist[0];
        QString conf_msg=conflist[1];
        if(conf_flag==flag){
            result=conf_msg;
            break;
        }
    }
    file.close(); // 关闭文件
    return result;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

deepallin

打个赏吧,三瓜两枣不嫌弃^_^

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值