Qt总结之二十三:QByteArray与char、int、float(及其数组)之间的互相转化

前言


因项目中,进程间通信使用UDP+TCP双通信方式,去传输大量数据(nTB数据),并在前端界面用OpenGL显示并绘制其波形。为防止丢帧导致图形出现波动,这里采用TCP确保数能够正常显示且刷新频率稳定。客户端和服务端之间通信,

UDP中:

发送:

qint64    writeDatagram(const char * data, qint64 size, const QHostAddress & address, quint16 port)
qint64    writeDatagram(const QByteArray & datagram, const QHostAddress & host, quint16 port)

接收:

qint64    readDatagram(char * data, qint64 maxSize, QHostAddress * address = 0, quint16 * port = 0)

TCP中:
发送:

qint64 write(const char *data, qint64 len);
qint64 write(const char *data);
inline qint64 write(const QByteArray &data)


接收:

qint64 read(char *data, qint64 maxlen);
QByteArray read(qint64 maxlen);
QByteArray readAll();

会发现,不管是TCP 还是UDP,Qt都提供了两种参数方式进行传值。

 

去保存一段定长的char型数组,里面可能有\0等字符,所以当作字符串varchar处理铁定丢失数据,所以要用二进制保存BLOB,这样对应的Qt数据类型要用QByteArray进行处理,原来只用到QByteArray转换成 char* 类型

2.QByteArray与char*的转换


2.1 QByteArray 转 char*
方式1 传统方式data()和size()函数 (方便)
 

QByteArray array(10, 'Q');//初始化
//array 赋值等代码
//...

// 转化
char *buf;//只是一个指针
int len;//buf的长度
buf = array.data();
len = array.size();


方式2 memcpy()方式 (灵活)
 

QByteArray array(9,'Q');
char buf[10];//数组
int len_array = array.size();
int len_buf = sizeof(buf);
int len = qMin( len_array, len_buf );

// 转化
memcpy( buf, array,  len );


2.2 char* 转 QByteArray
方法1 利用构造函数 (方便)

char buf[10];
//给buf赋值
for (int i = 0; i < 10; i++)
{
    buf[i] = (i + 1) % 3;//其中存在'\0'元素
}

// 转化
QByteArray array;
array = QByteArray(buf, 10);//因为buf[]中有`\0`,必须要写上数据长度;否则,数据会直接截断,丢失数据


方式2 memcpy()方式 (灵活)

char buf[10];
//给buf赋值
for (int i = 0; i < 10; i++)
{
    buf[i] = (i + 1) % 3;//其中存在'\0'元素
}

// 转化
QByteArray array;
array.resize(sizeof(buf));//重置数据大小
memcpy(array.data(), buf, sizeof(buf));//copy数据


3.QByteArray与int 以及int[] 的转换
3.1. int 与 QByteArray 互转

[1] int 转 QByteArray

// int 转 QByteArray
int  intVar = 199;

QByteArray array;
int len_intVar = sizeof(intVar);
array.resize(len_intVar);
memcpy(array.data(), &intVar, len_intVar);



[2]QByteArray 转 int

// QByteArray 转 int
// array 数据接上面
int  outIntVar;
memcpy(&outIntVar, array.data(), len_intVar);
//memcpy(&outIntVar, array, len_intVar);//此行代码与上句通用



3.2. int[] 与 QByteArray 互转
[1] int[] 转 QByteArray

// int[] 转 QByteArray
// int[] 转 QByteArray
int  intVar[4] = {1,2,9,0};//初始化变量赋值

QByteArray array;
int len_intVar = sizeof(intVar);
array.resize(len_intVar);
//转换 int[] -> QByteArray
memcpy(array.data(), &intVar, len_intVar);



[2]QByteArray 转 int[]
 

// QByteArray 转 int[]
// array 数据接上面
int  outIntVar[4];
memcpy(&outIntVar, array.data(), len_intVar);
//memcpy(&outIntVar, array, len_intVar);//此行代码与上句通用


4.QByteArray与float 以及float[] 的转换
其实完全可以参考第3节,int的用法.

4.1. float[] 与 QByteArray 互转
[1] float[] 转 QByteArray

// float[] 转 QByteArray
float  fVar[4] = { 1.1, 2.3, 9.5, 0.2 };//初始化变量赋值

QByteArray array;
int len_fVar = sizeof(fVar); // 4*4 = 16 (一个float占4个字节)
array.resize(len_intVar);
memcpy(array.data(), &fVar, len_fVar);



[2]QByteArray 转 float[]

//  QByteArray 转 float[]
float  outFvar[4];
memcpy(&outIntVar, array.data(), len_fVar);
//memcpy(&outFvar, array, len_fVar);//此行代码与上句通用



4.2. float 与 QByteArray 互转


可以安全参考int。

Qt中,如果你有一个`QByteArray`(字节数组)存储了十六进制数据,并想要将其换为带符号的浮点数(如`float`),你需要先解析十六进制字符串,然后根据其长度和前缀(如果是单精度或双精度)来换。 例如,对于单精度(32位)浮点数,你可以按照这样的步骤操作: 1. 将十六进制字符串分为两个部分,前4个字符代表小数点前的部分,后8个字符(如果存在的话)代表小数点后的部分。 2. 分别将这两个部分换成对应的无符号整数。 3. 对于小数部分,通常需要乘以16的相应次方(比如`0x1p+X`格式中的`X`),因为浮点数的小数部分是以指数形式表示的。 4. 合并这两个整数,将有符号整数换为`float`,注意正负符号取决于十六进制的第一个字符(对于小端模式的系统,这通常是第一位)。 如果涉及到双精度(64位)浮点数,处理过程类似,只是需要处理更多的字节。 以下是一个简单的示例,假设你已经有了一个名为`hexString`的`QByteArray`: ```cpp #include <QString> #include <QByteArray> #include <QLatin1Char> #include <QDataStream> // ... (假设已经读取了十六进制字符串到hexString) // 拆分十六进制字符串 QString intPart = hexString.left(8); QString fracPart = hexString.mid(8); // 换为无符号整数 quint32 uintIntPart = qint32(intPart.toHex().toUpper()); quint32 uintFracPart = qint32(fracPart.remove('p').toHex().toUpper()); // 计算指数和整数部分 int exponent = QByteArray{fracPart}.toInt(); quint64 floatNum = quint64(uintIntPart) << exponent; // 如果是双精度,这里会需要处理另外8位 if (hexString.length() > 16) { // ... } // 将数值换为带符号float bool isNegative = hexString.at(0) == QLatin1Char('-'); float signedFloat = static_cast<float>(isNegative ? -floatNum : floatNum); ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值