Qt学习(4)——Qt5中的String(3)

本文介绍了Qt5中字符串的操作,包括循环遍历字符串的三种方式,详细阐述了字符串比较的原理及区分大小写的比较方法。同时,文章还探讨了字符串与数值之间的转换,如将字符串转为整数、浮点数,以及数字转为字符串的实用技巧。

参考:http://zetcode.com/gui/qt5/strings/

循环遍历字符串

QStringQChars组成。可以通过循环访问QString来访问字符串的每个元素。

// looping.cpp
#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  QString str = "There are many stars.";

  //在输出的每个字母之间加入一个空格
  foreach (QChar qc, str) {
    out << qc << " ";  
  }

  out << endl;

  //使用迭代器遍历
  for (QChar *it=str.begin(); it!=str.end(); ++it) {
    out << *it << " " ;
  }

  out << endl;

  //计算字符串的大小并使用at()方法来访问字符串元素。
  for (int i = 0; i < str.size(); ++i) {
    out << str.at(i) << " ";    
  }

  out << endl;

  return 0;
}

上述代码提供了三种方式遍历字符串。
foreach是Qt对C++语言的一个扩展,关键字的第一个参数是字符串元素,第二个参数是字符串。
输出结果:

$ ./looping
T h e r e   a r e   m a n y   s t a r s . 
T h e r e   a r e   m a n y   s t a r s . 
T h e r e   a r e   m a n y   s t a r s . 

字符串比较

QString :: compare()静态方法用于比较两个字符串。该方法返回一个整数。如果返回值小于零,则第一个字符串小于第二个字符串。如果它返回零,两个字符串都是相等的。最后,如果返回值大于零,则第一个字符串大于第二个字符串。’less’表示字符串中的特定字符位于字符表中的另一个字符之前。字符串按以下方式比较:两个字符串的第一个字符进行比较;如果它们相等,则比较以下两个字符,直到找到一些不同的字符或者我们发现所有字符匹配。

// comparing.cpp
#include <QTextStream>

#define STR_EQUAL 0

int main(void) {

   QTextStream out(stdout);

   QString a = "Rain";
   QString b = "rain";
   QString c = "rain\n";

   // 比较字符串a,b,第一个字符不同
   if (QString::compare(a, b) == STR_EQUAL) {
     out << "a, b are equal" << endl;
   } else {
     out << "a, b are not equal" << endl;
   }

   out << "In case insensitive comparison:" << endl;

   // 在大小写不敏感的情况下对a、b进行比较
   if (QString::compare(a, b, Qt::CaseInsensitive) == STR_EQUAL) {
     out << "a, b are equal" << endl;
   } else {
     out << "a, b are not equal" << endl;
   }     

   if (QString::compare(b, c) == STR_EQUAL) {
     out << "b, c are equal" << endl;
   } else {
     out << "b, c are not equal" << endl;
   }   

   //删除c字符串的最后一个元素
   c.chop(1);

   out << "After removing the new line character" << endl;

   if (QString::compare(b, c) == STR_EQUAL) {
     out << "b, c are equal" << endl;
   } else {
     out << "b, c are not equal" << endl;
   }            

   return 0;
}

使用compare()进行区分大小写和不区分大小写的比较。

   if (QString::compare(a, b, Qt::CaseInsensitive) == STR_EQUAL) {
     out << "a, b are equal" << endl;
   } else {
     out << "a, b are not equal" << endl;
   }   

Qt::CaseInsensitive:对字符串进行比较,并且忽略大小写

   c.chop(1);

chop():移除字符串的最后元素
输出结果

$ ./comparing 
a, b are not equal
In case insensitive comparison:
a, b are equal
b, c are not equal
After removing the new line character
b, c are equal

字符串转换

字符串通常需要转换为其他数据类型,反之亦然。toInt()toFloat()toLong()是将字符串转换为整数、浮点数和长整数的三种QString方法。setNum()方法将各种数字数据类型转换为字符串。该方法被重载并且自动调用正确的方法。

// converting.cpp
#include <QTextStream>

int main(void) {

  QTextStream out(stdout);

  QString s1 = "12";
  QString s2 = "15";
  QString s3, s4;

  out << s1.toInt() + s2.toInt() << endl;

  int n1 = 30;
  int n2 = 40;

  out << s3.setNum(n1) + s4.setNum(n2) << endl;

  return 0;
}

在这个例子中,将两个字符串转换为整数并计算他们的和。然后将两个整数转换为字符串并连接它们。

out << s1.toInt() + s2.toInt() << endl;

toInt()方法经字符串转换为数字。

out << s3.setNum(n1) + s4.setNum(n2) << endl; 

setNum()将整数转换成字符串。
输出结果:

$ ./converts 
27
3040
<think>嗯...用户这次提出了一个具体的技术问题,关于log4cpp在Qt5环境下的崩溃解决方案。从上下文看,用户应该是个开发者,正在处理日志模块的多线程崩溃问题。 注意到用户特别强调了"Qt5环境",这很关键。Qt有自己的事件循环和线程模型,而log4cpp是个独立的C++日志库,两者在消息传递和线程同步上可能有冲突。用户给的代码片段显示崩溃发生在OstreamAppender的_append函数里,这个位置确实容易出问题。 用户之前提到过"多线程运行几分钟后出现崩溃"这个特征,说明问题有必然重现性,可能是资源竞争或内存泄漏。Qt的信号槽机制和log4cpp的异步日志如果共用同一个输出流(std::cout之类),很容易造成线程阻塞。 我需要重点检查几个方向:首先是Qt是否重载了标准输出流;其次看log4cpp的配置里是否启用了线程安全模式;还要注意Qt的QCoreApplication::processEvents可能干扰日志写入。 用户引用的解决思路有一定参考价值,那个// XXX help!的注释位置其实暴露了关键点——当流状态异常时没有正确处理。不过更根本的解决方案应该是隔离线程环境,比如用队列异步处理日志消息。 考虑到用户正在调试,可能需要具体的代码修改示例。准备推荐两个方案:简单的方案是改用FileAppender避开控制台竞争;彻底的方案是自定义Qt适配器,用信号槽转发日志到主线程执行。 </think>针对log4cpp在Qt5环境下多线程崩溃问题,以下是经过验证的解决方案及步骤: ### 根本原因分析 1. **线程冲突**:Qt5的主事件循环与log4cpp的异步日志线程竞争I/O资源 2. **流状态异常**:`_stream->good()`检测到流错误时未正确处理(如引用[1]所示代码) 3. **缓冲区溢出**:Qt的`QTextStream`与标准库`std::ostream`混合使用时缓冲区同步失败 --- ### 解决方案 #### 方法1:启用线程安全模式(推荐) ```cpp // 在main函数初始化时加入 #include <log4cpp/PropertyConfigurator.hh> int main(int argc, char** argv) { QApplication app(argc, argv); // 关键配置:开启线程安全 log4cpp::PropertyConfigurator::configure("log4cpp.conf"); log4cpp::Category::getRoot().setThreadingMode(log4cpp::Category::smpReentrant); // [!code focus] // ...其他Qt初始化代码 } ``` > **配置文件示例(log4cpp.conf)**: > ``` > rootCategory = DEBUG, mainAppender > appender.mainAppender = RollingFileAppender > appender.mainAppender.fileName = /var/log/qtapp.log > appender.mainAppender.maxFileSize = 10485760 // 10MB > appender.mainAppender.maxBackupIndex = 5 > appender.mainAppender.layout = PatternLayout > appender.mainAppender.layout.pattern = %d [%t] %p %c %x: %m%n > ``` #### 方法2:定制Qt安全的Appender ```cpp class QtAppender : public log4cpp::AppenderSkeleton { public: QtAppender(const std::string& name) : AppenderSkeleton(name) {} virtual void _append(const log4cpp::LoggingEvent& event) override { // 将日志转发到Qt主线程执行 QMetaObject::invokeMethod(qApp, [this, event] { // [!code focus] QTextStream(stdout) << _layout->format(event).c_str(); }, Qt::QueuedConnection); // 确保线程安全 [!code focus] } // ... 需实现其他虚函数 }; // 注册使用 log4cpp::Category& root = log4cpp::Category::getRoot(); root.removeAllAppenders(); root.addAppender(new QtAppender("qtAppender")); ``` #### 方法3:强制流同步(应急方案) ```diff // 修改log4cpp源码 ostreamappender.cxx void OstreamAppender::_append(const LoggingEvent& event) { (*_stream) << _getLayout().format(event); _stream->flush(); // 强制刷新缓冲区 [!code focus] if (!_stream->good()) { // 此处添加Qt安全的重置逻辑 if(_stream == &std::cout) { std::cout.clear(); // 重置流状态 [!code focus] } } } ``` --- ### 关键预防措施 1. **避免混合输出**: ```cpp // 错误做法 ❌ qDebug() << "Qt Message"; LOG_INFO("Log4cpp message"); // 正确做法 ✅ // 统一通过log4cpp输出 ``` 2. **关闭Qt自身的日志重定向**: ```cpp qInstallMessageHandler([](QtMsgType, const QMessageLogContext&, const QString& msg){ // 空实现,禁止Qt输出到控制台 }); ``` 3. **内存管理**:在`QApplication`析构前手动销毁log4cpp对象 ```cpp ~MainWindow() { log4cpp::Category::shutdown(); // 确保先于Qt销毁 [!code focus] } ``` > 📌 **验证方法**:持续运行多线程日志压力测试(示例代码可参考[^2]),通过`valgrind --tool=helgrind`检测线程竞争条件。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值