第一章QT基本数据类型
文章目录
- 第一章QT基本数据类型
- 1.基础类型
- 2. log输出
- 基本分类
- 格式化日志
- 普通格式化
- 条件格式化
- 环境变量格式化
- 禁用输出
- 3.字符串类型
1.基础类型
Qt基本数据类型定义在#include <QtGlobal>
中,Qt基本数据类型有:
类型名称 | 注释 | 备注 |
---|---|---|
qint8 | signed char | 字符型 |
qint16 | signed short | 短整型 |
qint32 | signed int | 整型 |
qint64 | long long int | 长整型 |
qintptr | qint32或qint64 | 指针类型,32位系统是qint32~ |
qlonglong | long long int或(__int64) | Windows中定义为__int64 |
qptrdiff | qint32或qint64 | 指针偏移,根据系统类型不同而不同~ |
qreal | double或float | 实型,除非配置了-qreal float选项,否则默认double |
quint8 | unsigned char | 无符号字符型 |
quint16 | unsigned short | 无符号短整型 |
quint32 | ~ | ~ |
quint64 | ~ | ~ |
quintptr | ~ | ~ |
qulonglong | ~ | ~ |
uchar | unsigned char | ~ |
uint | unsigned int | ~ |
ulong | unsigned long | ~ |
ushort | unsigned short | ~ |
qsizetype | size_t | ~ |
2. log输出
在Qt中进行log输出,Qt框架提供了专门用于日志输出的类,头文件名QDebug
基本分类
- qDebug:调试信息提示
- qInfo:输出信息
- qWarning:一般的警告提示
- qCritical:严重的错误提示
- qFatal:致命错误提示,会直接中断程序
qint8 v1='c';qint32 v2=100;quint8 v3='a';quint32 v4=100;qintptr *p=nullptr;qDebug("v1 = %" PRIdQSIZETYPE "",v1);qInfo("v2 = %d",v2);qWarning("v3=%c",v3);qCritical("v4=%d",v4);qFatal("致命错误");
格式化日志
默认情况下,日志格式是只输出对应的日志内容没有额外信息的。我们可以通过修改环境变量QT_MESSAGE_PATTERN或者调用方法qSetMessagePattern来修改日志的输出格式,日志格式中常用的占位符符号如下:
%{appname} 应用程序的名称(QCoreApplication::applicationName())
%{category} 日志所处的领域
%{file} 打印该日志的文件路径
%{fuction} 打印日志的函数
%{line} 打印日志在文件中的行数
%{message} 打印日志的内容
%{pid} 打印日志程序的PID
%{threadid} 打印日志的线程ID
%{qthreadptr} 打印日志的线程指针
%{type} 日志级别("debug","warning","critical" or "fatal")
%{time process} 日志发生时程序启动了多久
%{time boot} 日志发生时系统启动了多久
%{time [format]}以固定时间格式输出日志打印的时间,默认为QISODate格式
普通格式化
qSetMessagePattern("%{time [yy-MM-dd hh:mm:ss]} %{file} %{function}(%{line}) %{message}");qInfo()<<"qInfo";qDebug()<<"qDebug";qWarning()<<"qWarning";qCritical()<<"qCritical";
注意:在Release模式下,文件名、函数名、行号获取不到,需要添加编译时宏QT_MESSAGELOGCONTEXT
添加方法如下:
在cmakeLists.txt文件中添加
add_definitions(-DQT_MESSAGELOGCONTEXT) # 全局范围可用
或者
target_compile_definitions(你的可执行程序名称 PRIVATE QT_MESSAGELOGCONTEXT) #目标范围可用
条件格式化
QString s_debug="%{if-debug} [%{type}] line%{line} %{endif}";QString s_info = "%{if-info} [%{type}] %{file} %{endif}";qSetMessagePattern("%{time [yy-MM-dd hh:mm:ss]}"+s_debug+s_info+"%{message}");
- 案例把日志写入文件
#include "MainWindow.h"
#include <QtGlobal>
#include <QApplication>
#include <QDebug>
#include <QFile>//日志消息处理函数
void logMessageHander(QtMsgType type,const QMessageLogContext &ctx,const QString &msg)
{//获取格式化日志信息QString typestr=qFormatLogMessage(type,ctx,msg);//可以根据日志级别进行过滤(比如不要debug,可以直接return)QString levelText;switch(type){case QtDebugMsg:levelText="debug";return;break;case QtWarningMsg:levelText ="warning";break;case QtCriticalMsg:levelText ="critical";break;case QtFatalMsg:levelText ="fatal";break;case QtInfoMsg:levelText ="info";break;}QFile file("mylog.txt");file.open(QIODeviceBase::WriteOnly|QIODeviceBase::Append);if(file.isOpen()){QTextStream tStream(&file);tStream<<typestr<<"\n";file.close();}
}
int main(int argc, char *argv[])
{QApplication a(argc, argv);QString s_debug="%{if-debug} [%{type}] line%{line} %{endif}";QString s_info = "%{if-info} [%{type}] %{file} %{endif}";qSetMessagePattern("%{time [yy-MM-dd hh:mm:ss]}"+s_debug+s_info+"%{message}");qInstallMessageHandler(logMessageHander);MainWindow w;qint8 v1='c';qint32 v2=100;quint8 v3='a';quint32 v4=100;qintptr *p=nullptr;QString say="你好,世界";qDebug("v1 = %" PRIdQSIZETYPE "",v1);qDebug()<<say;qInfo("v2 = %d",v2);qWarning("v3=%c",v3);qCritical("v4=%d",v4);//qFatal("致命错误");w.show();return a.exec();
}
环境变量格式化
qputenv("QT_MESSAGE_PATTERN",QByteArray("%{time yyyyMMdd h:mm:ss.zzz t} %{if-debug}D %{endif}""%{file} %{function}(%{line}) %{message}"));
和上面的qSetMessagePattern()一起使用的话,qputenv()就无效了
禁用输出
在开发或者调试时,我们必须借助日志来进行判断,但是当程序需要发布时,调试的日志信息不再需要,此时如果把代码删除,又不太方便
Qt提供了禁用qInfo、qwarning、qdebug输出的宏,qcritical和qfatal不能屏蔽
在CMakeLists.txt文件中添加如下三行,即可禁用
add_compile_definitions(QT_NO_INFO_OUTPUT)
add_compile_definitions(QT_NO_WARNING_OUTPUT)
add_compile_definitions(QT_NO_DEBUG_OUTPUT)
3.字符串类型
- QByteArray
可以通过这个类的构造函数申请一个内存,用于存储我们需要处理的字符串数据
- 构造函数
QByteArray arr1("hello,world!");QByteArray arr2("你好世界");QByteArray arr3(10,'a');qDebug()<<arr1;qDebug()<<QString::fromUtf8(arr2);//输出中文要通过qstring转qDebug()<<arr3;
- 数据操作
arr2.append("我好帅");//尾插qDebug()<<QString::fromUtf8(arr2); //"你好世界我好帅"arr1.insert(3,"abc");qDebug()<<arr1; //"helabclo,world!"arr1.insert(10,"@@&^",1);qDebug()<<arr1; //"helabclo,w@orld!"arr3.insert(20,'*');//超出长度用空格填充qDebug()<<arr3; //"aaaaaaaaaa *" arr1.remove(3,3); //移除第三个位置的后面3个字符qDebug()<<arr1; //"hello,w@orld!"arr1.removeIf([](char a){return a<'i';});//移除里面小于'i'的字符qDebug()<<arr1; //"lloworl"arr3.truncate(5);//从第5个位置开始截断qDebug()<<arr3;//"aaaaa"arr3.replace(1,2,"我是个大聪明啊,你是不是也和我一样");//从第一个位置开始,把后面2个字符替换成内容qDebug()<<QString::fromUtf8(arr3);//"a我是个大聪明啊,你是不是也和我一样aa"
- 字符串查找和判断
qDebug()<<"arr3包含大聪明"<<arr3.contains("大聪明");//arr3包含大聪明 trueqDebug()<<"arr3以大聪明开头"<<arr3.startsWith("大聪明");//arr3以大聪明开头 falseqDebug()<<"arr3以大聪明结尾"<<arr3.endsWith("大聪明");//arr3以大聪明结尾 false qDebug()<<arr1.at(2);qDebug()<<arr1[2];//第二个位置的字符
- 遍历
for(auto i :arr1)qDebug()<<i;
- QString
- 构造函数
QString s1="你好";QString s2(QByteArray("abc"));qDebug()<<s1<<s2;
- 数据操作
和QByteArray类似,append,prepend,insert
s2.chop(1);//尾部删除一个字符qDebug()<<s2;
- 字符串格式化
s2 = QString::asprintf("hello %s","world");//跟sprintf差不多qDebug()<<s2;qDebug()<<QString("我叫%1,今年%2岁,职业是%3").arg("张三").arg(30).arg("程序员");//"我叫张三,今年30岁,职业是程序员"
- 不同字符串类型转换
//string->qstrings1= QString::fromStdString("你好china");qDebug()<<s1;//qstring->stringstd::string str1= s1.toStdString();std::cout<<str1<<std::endl;//qstring->qbytearryQByteArray qarr1 = s1.toUtf8();//转换为utf8编码格式的字符串qarr1 = s1.toLatin1();//转换为latin1编码,不支持中文qarr1 = s1.toLocal8Bit();//转换为本地编码,跟随操作系统
- QVariant
变体数据类型,可以保存很多的数据类型,包括QBrush,QColor,QCursor,QDateTime,QFont,QKeySequence,QPalette,QPen,QPixmap,QPoint,QRect,QRegion,QSize和QString,并且还有C++基本类型,如int,float等
QVariant var1(666);QVariant var2 = QVariant::fromValue(QString("野人诺"));QVariant var3;var3.setValue(QByteArray("张三"));qDebug()<<var1.toInt()<<var2.toString()<<var3.toByteArray();
qDebug()输出时,转换类型要和原来的一致,否则输出为空
上面运行结果为: 666 “野人诺” “”
qDebug()<<var3.typeName();//const char*
自定义类型
mainwindow.h加上自定义类
class Person
{
public:QString name;quint8 age;friend QDebug operator<<(QDebug debug,const Person &person){debug<<"Person("<<person.name<<","<<person.age<<")";return debug;}
};
Q_DECLARE_METATYPE(Person)
main.cpp里使用
var3.setValue(Person());qDebug()<<var3.value<Person>();//输出结果Person( "" , 0 )return a.exec();
判断类型使用
if(var3.canConvert<int>()){qDebug()<<var3.value<int>();}else if(var3.canConvert<Person>()){qDebug()<<var3.value<Person>();}elseqDebug()<<var3.typeName();return a.exec();