【转】Qt中的QString,QByteArray,Qchar, char*

本文详细介绍了Qt框架中的QString类,包括其特点、如何使用以及与C风格字符串和std::string的区别。探讨了QString的各种操作方法,如连接、格式化、查找、替换等。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

先要说的是QString。

      之所以把QString单独拿出来,是因为string是很常用的一个数据结构,甚至在很多语言中,比如JavaScript,都是把string作为一种同int等一样的基本数据结构来实现的。

      每一个GUI程序都需要string,这些string可以用在界面上的提示语,也可以用作一般的数据结构。C++语言提供了两种字符串的实现:C风格的 字符串,以'\0‘结尾;std::string,即标准模版库中的类。Qt则提供了自己的字符串实现:QString。QString以16位 Uniode进行编码。我们平常用的ASCII等一些编码集都作为Unicode编码的子集提供。关于编码的问题,我们会到以后的时候再详细说明。

在使用QString的时候,我们不需要担心内存分配以及关于'\0'结尾的这些注意事项。QString会把这些问题解决。通常,你可以把 QString看作是一个QChar的向量。另外,与C风格的字符串不同,QString中间是可以包含'\0'符号的,而length()函数则会返回 整个字符串的长度,而不仅仅是从开始到'\0'的长度。

     同Java的String类类似,QString也重载的+和+=运算符。这两个运算符可以把两个字符串连接到一起,正像Java里面的操作一样。QString可以自动的对占用内存空间进行扩充,这种连接操作是恨迅速的。下面是这两个操作符的使用:

  1. QString str = "User: "  ;  
  2. str += userName + "\n"  ; 

QString的append()函数则提供了类似的操作,例如:

  1. str = "User: "  ;  
  2. str.append(userName);  
  3. str.append("\n"  ); 

C语言中有printf()函数作为格式化输出,QString则提供了一个sprintf()函数实现了相同的功能:

  1. str.sprintf("%s %.1f%%"  , "perfect competition"  , 100.0); 

这句代码将输出:perfect competition 100.0%,同C语言的printf()一样。不过前面我们也见到了Qt提供的另一种格式化字符串输出的函数arg():

  1. str = QString("%1 %2 (%3s-%4s)"  )  
  2.       .arg("permissive"  ).arg("society"  ).arg(1950).arg(1970); 

这段代码中,%1, %2, %3, %4作为占位符,将被后面的arg()函数中的内容依次替换,比如%1将被替换成permissive,%2将被替换成society,%3将被替换成 1950,%4将被替换曾1970,最后,这句代码输出为:permissive society (1950s-1970s). arg()函数比起sprintf()来是类型安全的,同时它也接受多种的数据类型作为参数,因此建议使用arg()函数而不是传统的 sprintf()。

使用static的函数number()可以把数字转换成字符串。例如:

  1. QString str = QString::number(54.3); 

你也可以使用非static函数setNum()来实现相同的目的:

  1. QString str;  
  2. str.setNum(54.3); 

而一系列的to函数则可以将字符串转换成其他基本类型,例如toInt(), toDouble(), toLong()等。这些函数都接受一个bool指针作为参数,函数结束之后将根据是否转换成功设置为true或者false:

  1. bool    ok;  
  2. double    d = str.toDouble(&ok);  
  3. if   (ok)  
  4. {  
  5.     // do something...   
  6. } else    {  
  7.     // do something...   

对于QString,Qt提供了很多操作函数,例如,使用mid()函数截取子串:

  1. QString x   = "Nine pineapples"  ;  
  2. QString y   = x  .mid(5, 4);            // y   == "pine"  
  3. QString z   = x  .mid(5);               // z   == "pineapples" 

mid()函数接受两个参数,第一个是起始位置,第二个是取串的长度。如果省略第二个参数,则会从起始位置截取到末尾。正如上面的例子显示的那样。

函数left()和rigt()类似,都接受一个int类型的参数n,都是对字符串进行截取。不同之处在于,left()函数从左侧截取n个字符,而right()从右侧开始截取。下面是left()的例子:

  1. QString x = "Pineapple"  ;  
  2. QString y = x.left(4);      // y == "Pine"   

函数indexOf()返回字符串的位置,如:

  1. QString x = "sticky question"  ;  
  2. QString y = "sti"  ;  
  3. x.indexOf(y);               // returns 0   
  4. x.indexOf(y, 1);            // returns 10   
  5. x.indexOf(y, 10);           // returns 10   
  6. x.indexOf(y, 11);           // returns -1   

函数startsWith()和endsWith()可以检测字符串是不是以某个特定的串开始或结尾,例如:

  1. if    (url.startsWith("http:"  ) && url.endsWith(".png"  ))  
  2. {  

这段代码等价于

  1. if    (url.left(5) == "http:"   && url.right(4) == ".png"  )  
  2. {  

不过,前者要比后者更加清楚简洁,并且性能也更快一些。

QString还提供了replace()函数供实现字符串的替换功能;trimmed()函数去除字符串两侧的空白字符(注意,空白字符包括空 格、Tab以及换行符,而不仅仅是空格);toLower()和toUpper()函数会将字符串转换成小写大写字符串;remove()和 insert()函数提供了删除和插入字符串的能力;simplified()函数可以将串中的所有连续的空白字符替换成一个,并且把两端的空白字符去 除,例如"   \t   ”会返回一个空格" "。

将const char *类型的C风格字符串转换成QString也是很常见的需求,简单来说,QString的+=即可完成这个功能:

  1. str += " (1870)"  ; 

这里,我们将const char * 类型的字符串" (1870)"转换成为QString类型。如果需要显式的转换,可以使用QString的强制转换操作,或者是使用函数fromAscii()等。为了 将QString类型转成const char *字符串,需要进行两步操作,一是使用toAscii()获得一个QByteArray类型对象,然后调用它的data()或者constData()函 数,例如:

  1. printf("User: %s\n"  , str.toAscii().data()); 

为了方便使用,Qt提供了一个宏qPrintable(),这个宏等价于toAscii().constData(),例如:

  1. printf("User: %s\n"  , qPrintable(str)); 

我们调用QByteArray类上面的data()或者constData()函数,将获得QByteArray内部的一个const char*类型的字符串,因此,我们不需要担心内存泄漏等的问题,Qt会替我们管理好内存。不过这也暗示我们,注意不要使用这个指针太长时间,因为如果 QByteArray被delete,那么这个指针也就成为野指针了。如果这个QByteArray对象没有被放在一个变量中,那么当语句结束 后,QbyteArray对象就会被delete,这个指针也就被delete 了。

接下来说说QString和QByteArray之间的转换  ,这里说两种,都是从网上搜到的。

第一种,数据流的方式,这里只说从QByteArray转向QString。

QByteArray encodedString = "xxx";
QTextCodec *codec = QTextCodec::codecForName("KOI8-R");
QString string = codec->toUnicode(encodedString);

[另外]

char ch_str[10] = "搜索";

QString str = QString::fromLocal8Bit(ch_str);

第二种,常用的,网上看到的。

//常用参数类型:char *字符串, QByteArray字符数组, QString字符串
//需要转换:char * ---转---  QByteArray  ---需要调用QByteArray类的构造函数
char*     --------------QByteArray(const char*)-----------> QByteArray
char* str;
QByteArray byte(str);

char*     --------------QString(const char*)-------------->QString
char* str;
QString string(str);

QByteArray   -------------data()-------------------->char*
QByteArray byte;
char* str = byte.data();

QByteArray  ------------QString()------------------->QString
QByteArray byte;
QString string(byte);

QString  --------------toAscii()--------------------> QByteArray
QString string;
QByteArray byte = string.toAscii();

QString -------------qPrintable()------------------>char*
QString string;
char* str = qPrintable(string);

或QString --------toLatin1()------QByteArray----------data()--------->char*

QString string;

QByteArray ba=string.toLatin1();

const char* str = ba.data();

<think>好的,我现在需要帮助用户解决QtQStringQByteArray的问题。首先,我得回忆一下用户之前的问题,他们之前问过QStringList的赋值,现在向了QStringQByteArray之间的换。这说明用户可能在处理字符串数据的存储、传输或者编码相关的问题。 接下来,我需要确定用户的具体需求。QStringQByteArray通常发生在需要将Unicode字符串换为字节流的时候,比如网络传输、文件读写或者与需要字节数组的API交互。用户可能遇到了编码问题,或者在处理不同字符集的数据换。 然后,我要考虑Qt中提供的换方法。最基本的是使用QString的toUtf8()、toLatin1()等方法,这些函数返回QByteArray。需要解释不同编码方法的区别,比如UTF-8Latin-1的应用场景,以及可能的数据丢失问题。 用户可能对编码不太熟悉,所以需要简单说明编码的重要性。例如,如果字符串包含非拉丁字符(如中文),使用Latin1会导致数据丢失,而UTF-8则能保留所有字符。这时候可能需要提醒用户根据具体需求选择合适的编码方式。 另外,用户可能不知道换后的QByteArray如何再QString,所以在回答中包含双向换的例子会更有帮助。比如使用QByteArray的data()获取指针,或者使用QString的fromUtf8()等方法。 还要考虑特殊场景,比如处理二进制数据或包含空字符的字符串。这时候需要使用toLocal8Bit()或者确保正确的中断处理,避免截断数据。此外,性能优化也是一个点,比如预分配内存或避免不必要的换,这对处理大量数据很重要。 用户可能遇到的常见错误包括直接使用char*导致的编码问题,或者未指定编码方式造成的乱码。需要提醒用户检查换后的数据是否正确,特别是在跨平台或不同语言环境下的表现。 最后,要确保回答结构清晰,分点列出不同方法,并给出示例代码,帮助用户逐步理解应用。同时,最佳实践注意事项能帮助用户避免常见陷阱,提高代码的健壮性。</think>以下是Qt中`QString`与`QByteArray`相互换的完整指南: --- ### 一、基础换方法 #### 1. **UTF-8编码换(推荐通用方案)** ```cpp // QStringQByteArray QString str = "你好Qt世界"; QByteArray utf8Data = str.toUtf8(); // 最常用方式,支持多语言 // QByteArrayQString QString restoredStr = QString::fromUtf8(utf8Data); ``` #### 2. **Latin1编码换(ASCII扩展)** ```cpp // 仅适用于纯英文字符 QString engStr = "Hello Qt"; QByteArray latinData = engStr.toLatin1(); // 编码范围0-255 QString backStr = QString::fromLatin1(latinData); // 注意中文会丢失! ``` --- ### 二、高级换场景 #### 1. **本地编码换(系统默认)** ```cpp // 依赖系统编码(Windows可能是GBK,Linux可能是UTF-8) QByteArray localData = str.toLocal8Bit(); QString localStr = QString::fromLocal8Bit(localData); ``` #### 2. **指定特定编码** ```cpp // 使用QTextCodec处理特殊编码(需包含头文件<QTextCodec>) QTextCodec *codec = QTextCodec::codecForName("GB2312"); QByteArray gbData = codec->fromUnicode(str); // 为GB2312 QString gbStr = codec->toUnicode(gbData); // 从GB2312回 ``` --- ### 三、二进制数据处理 #### 1. **原始字节访问** ```cpp // 直接访问底层数据指针 const char *rawData = utf8Data.constData(); int dataSize = utf8Data.size(); // 获取字节长度 ``` #### 2. **带空字符(NUL)的字符串** ```cpp QString specialStr = "ABC\0DEF"; // 包含空字符 QByteArray binData = specialStr.toUtf8(); binData.append('\0'); // 可追加空字符 // 换回QString需显式指定长度 QString recovered = QString::fromUtf8(binData.constData(), binData.size()); ``` --- ### 四、最佳实践建议 #### 1. **编码选择原则** - 跨平台/网络传输 → **始终使用UTF-8** - 处理纯英文 → Latin1更高效 - 遗留系统对接 → 使用对应编码(如GBK) #### 2. **错误处理技巧** ```cpp // 检查编码可用性 if(!QTextCodec::codecForName("Shift-JIS")) { qWarning() << "编码不支持"; } // 安全换示例 QByteArray safeConvert(const QString &str) { if(str.contains(QChar(0x2603))) { // 检查特殊字符 return str.toUtf8(); // 强制使用UTF-8 } return str.toLatin1(); } ``` --- ### 五、性能优化 #### 1. **预分配内存** ```cpp QByteArray buffer; buffer.reserve(1024); // 预分配1KB buffer.append(str.toUtf8()); ``` #### 2. **避免临时对象** ```cpp // 错误方式:产生临时QByteArray sendData(str.toUtf8().constData()); // 正确方式:保持对象生命周期 QByteArray temp = str.toUtf8(); sendData(temp.constData()); ``` --- ### 六、常见误区 #### 1. **直接使用char*** ```cpp // 危险操作:未考虑编码 const char* cstr = str.toStdString().c_str(); // 可能丢失数据 // 正确方式:显式指定编码 const char* safeCstr = str.toUtf8().constData(); ``` #### 2. **编码不一致导致乱码** ```cpp // 错误示例:混合编码 QByteArray data = str.toLatin1(); // 假设str含中文 QString wrongStr = QString::fromUtf8(data); // 显示乱码 ``` --- ### 七、调试技巧 #### 1. **十六进制查看** ```cpp qDebug() << "Hex:" << utf8Data.toHex(); // 输出类似"e4bda0e5a5bd" ``` #### 2. **编码检测** ```cpp // 自动检测编码(需Qt 5.12+) QString detectEncoding(QByteArray data) { QTextCodec::ConverterState state; QTextCodec *codec = QTextCodec::codecForUtfText(data, nullptr); return codec->toUnicode(data.constData(), data.size(), &state); } ``` --- 通过合理选择编码方式遵循这些实践准则,可以确保`QString`与`QByteArray`换的可靠性效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值