Qt ASCII转Unicode再转中文

Qt ASCII转Unicode再转中文


1. 前言

这里主要是读取二代身份证的时候的一个协议处理,目前接触了两种模块,协议有一些差异。这里举下解析过程的例子来说明一下这个转换。

2. 示例及转换过程

一般二代证的解析结果中名字占三十个字节,但是这个三十字节不同模块返回会有一些差异,因为直接结果是ASCII,比如汉字“郑”的转成ASCII码为:“\u90D1”,b部分模块是直接返回两个字节0x90和0xd1,但是一些模块是存储成4个字节,而且串口读取到的是16进制码,所以就成了0x44(D)、0x31(1)、0x39(9)、0x30(0),这两种情况下,从串口读取到数据到转换为实际汉字,比较绕,这里分别说明一下转换过程。

2.1 ASCII码形式

这种的有60个字节,可以转换成30字节,再转15字节,也可以直接根据字符将60字节4字节划分成15部分分别处理,将ASCII拆分回复,这种情况不好理解,但是容易看出来,我们通过QByteArray接收到串口数据后直接打印显示的就是ASCII,可以直接打印出“D190”,我们通过串口接收后打印:

QBytearray readBuff;
readBuff = serialPort.readAll();

qDebug()<<readBuff;

这时我们通过ASCII转中文的在线转换工具可以看出来结果:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-mr1Asygc-1632995585382)(C:\Users\admin\AppData\Roaming\Typora\typora-user-images\image-20210930171446284.png)]

然后提取出来这60个字节(以ASCII码看是30字节,但是每一个字符占2个字节,就像“D1”,实际D和1每个的存储都需要占用一个字节),然后使用如下方式转换成汉字,而且每两个ASCII字节组成一个汉字,返回的字节高低字节还需要互换:

QByteArray tmp;

for (int i = 0; i < 59; i++) {
    tmp[i] = g_IDNumCardInfo[beginOffset + i];
    tmp[i + 1] = g_IDNumCardInfo[beginOffset + i + 1];
    //        qDebug() << tmp[i] << tmp[i + 1];
}
qDebug() << tmp << tmp.size();

QString nameRes;
for (int i = 0; i < 15; i++) {
    bool ok;
    QByteArray byte;
    //高低字节互换
    byte[0] = tmp[2 + 4 * i];
    byte[1] = tmp[3 + 4 * i];
    byte[2] = tmp[0 + 4 * i];
    byte[3] = tmp[1 + 4 * i];
    //        qDebug() << "file:" << __FILE__ << "line:" << __LINE__ << byte;
    auto *unicode = new QChar[1];

    unicode[0] = byte.toInt(&ok, 16);
    nameRes.append(QString::fromRawData(unicode, 1));
}

qDebug() << "file:" << __FILE__ << "line:" << __LINE__ << "名字:" << nameRes;

2.2 16进制字节

这种的30个字节再组合成15个字节进行转换即可,这种的实际上相当于已经做了一层封装,将ASCII的两个字符字节组合成了一个字节,这种的存储的时候直接高第字节互换然后存储到ushort数组中,然后直接再转成字符串就可以了:

ushort name[15];

for(int i = 0; i < 15; i++) {
    name[i] = g_IDNumCardInfo[8+2*i+1] << 8;
    name[i] = name[i] + (g_IDNumCardInfo[8+2*i] & 0x00ff);

    qDebug()<<"file:"<<__FILE__<<"line:"<<__LINE__<<QString().sprintf("%02x",name[i]);
}

QString nameRes = QString::fromUtf16(name, 15);
qDebug()<<"file:"<<__FILE__<<"line:"<<__LINE__<< nameRes;

3. 最后

每两个字节代表 1 个汉字的 Unicode 码,低位在前,比如90D1两个字节,但是9、0、d、1这四个字符传输时根据情况可以两两组合分两字节传输(0x90,0xD1)也可以不组合分四个字节传输(可以以ASCII当成字符传0x39、0x30、0x44、0x31,还可以直接以16进制样式传0x09、0x00、0x0d、0x01)。

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

昵称系统有问题

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值