1、前言
说到Qt,不能不说到C++,这门伟大的语言。由于其面向对象的编程思想和陡峭的学习曲线,一开始学习起来非常吃力。Qt从QT4开始基本封装了不少C++的工具库和界面库,并且支持跨平台,这是它最大的优点。相比于古老的MFC和使用C#的WPF来讲,我更喜欢Qt来进行C++的界面设计,这也是我重新开始拾起Qt学习之路的缘由。框架
说完了感觉和学习的缘由,就开始介绍Qt5了。我这里只作简单的介绍,重要的仍是在学习过程当中体会Qt的编程逻辑和设计思想。
Qt,坦白来讲,并不仅是一个界面库,他是C++编程思想的集大成者。它是获得完善的C++应用程序框架。使用Qt,在必定程度上你得到的是一个“一站式”、“全方位”的解决方案,STL。string、XML、数据库、网络这些零散的功能都包含在Qt中,而且获得了封装,一共开发者使用。
跨平台GUI一般有三种实现策略,分别是:
- API 映射:API 映射是说,界面库使用同一套 API,将其映射到不一样的底层平台上面。大致至关于将不一样平台的 API 提取公共部分。好比说,将 Windows 平台上的按钮控件和 Mac OS 上的按钮组件都取名为 Button。当你使用 Button 时,若是在 Windows 平台上,则编译成按钮控件;若是在 Mac OS 上,则编译成按钮组件。这么作的好处是,全部组件都是原始平台自有的,外观和原平生台一致;缺点是,编写库代码的时候须要大量工做用于适配不一样平台,而且,只能提取相同部分的 API。好比 Mac OS 的文本框自带拼写检测,可是 Windows 上面没有,则不能提供该功能。这种策略的典型表明是 wxWidgets。这也是一个标准的 C++ 库,和 Qt 同样庞大。它的语法看上去和 MFC 相似,有大量的宏。听说,一个 MFC 程序员能够很容易的转换到 wxWidgets 上面来。
- API 模拟:前面提到,API 映射会“缺失”不一样平台的特定功能,而 API 模拟则是解决这一问题。不一样平台的有差别 API,将使用工具库本身的代码用于模拟出来。按照前面的例子,Mac OS 上的文本框有拼写检测,可是 Windows 的没有。那么,工具库本身提供一个拼写检测算法,让 Windows 的文本框也有相同的功能。API 模拟的典型表明是 wine ——一个 Linux 上面的 Windows 模拟器。它将大部分 Win32 API 在 Linux 上面模拟了出来,让 Linux 能够经过 wine 运行 Windows 程序。由此能够看出,API 模拟最大优势是,应用程序无需从新编译,便可运行到特定平台上。另一个例子是微软提供的 DirectX,这个开发库将屏蔽掉不一样显卡硬件所提供的具体功能。使用这个库,你无需担忧硬件之间的差别,若是有的显卡没有提供该种功能,SDK 会使用软件的方式加以实现。
- GUI 模拟:任何平台都提供了图形绘制函数,例如画点、画线、画面等。有些工具库利用这些基本函数,再不一样绘制出本身的组件,这就是 GUI 模拟。GUI 模拟的工做量无疑是很大的,由于须要使用最基本的绘图函数将全部组件画出来;而且这种绘制很难保证和原生组件如出一辙。可是,这一代价带来的优点是,能够很方便的修改组件的外观——只要修改组件绘制函数便可。不少跨平台的 GUI 库都是使用的这种策略,例如 gtk+(这是一个 C 语言的图形界面库。使用 C 语言很优雅地实现了面向对象程序设计。不过,这也一样带来了一个问题——使用大量的类型转换的宏来模拟多态,而且它的函数名通常都比较长,使用下划线分割单词,看上去和 Linux 一模一样。gtk+ 并非模拟的原生界面,而有它本身的风格,因此有时候就会和操做系统的界面格格不入。),Swing 以及咱们的 Qt。
Qt 和 wxWidgets 同样,也是一个标准的 C++ 库。可是它的语法相似于 Java 的 Swing,十分清晰,并且使用信号槽(signal/slot)机制,让程序看起来很明白——这也是不少人优先选择 Qt 的一个很重要的缘由。不过,所谓“成也萧何,败也萧何”。这种机制虽然很清楚,可是它所带来的后果是你须要使用 Qt 的 moc 对程序进行预处理,才可以再使用标准的 make 或者 nmake 进行正常的编译,而且信号槽的调用要比普通的函数调用慢大约一个数量级(Qt 4 文档中说明该数据,但 Qt 5 还没有有官方说明)。Qt 的界面也不是原生风格的,尽管 Qt 使用 style 机制十分巧妙地模拟了原生界面。另外值得一提的是,Qt 不只仅可以运行在桌面环境中,还能够运行在嵌入式平台以及手机平台。
Qt 初版于 1991 年由 Trolltech (奇趣科技)发布。后来在 2008 年,Nokia 斥资 1.5 亿美圆收购 TrollTech,将 Qt 应用于 Symbian 程序开发。2012 年 8 月 9 日,Nokia 将 Qt 以 400 万欧元的价格出售给 Digia。
伴随着 Qt,一直有两种受权协议:商业受权以及开源受权。在 Qt 的早期版本,商业受权包含一些开源受权不提供的组件,可是在近期版本则不存在这个问题。以往人们对 Qt 的开源受权多有诟病。早期版本的 Qt 使用与 GPL 不兼容的协议受权,这直接致使了 KDE 与 GNOME 的战争(因为 Linux 使用 GPL 协议发布,GPL 协议具备传染性,做为 Linux 桌面环境的 KDE 倒是基于与 GPL 不兼容的 Qt 开发,这就不遵照 GPL 协议)。不过,如今 Qt 的开源版本使用的是 GPLv3 以及 LGPL 协议。这意味着,你能够将 Qt 做为一个库链接到一个闭源软件里面。能够说,Qt 协议的争议已经不存在了。
2、Qt的安装与配置
2.1 Qt的安装
上面是两种Qt,第一种是使用mingw编译和调试开发的Qt,第二种(红线中)是使用msvc也就是vs调试和编译的Qt。开始学习的建议安装第一种,自带gcc编译器和gdb调试器。安装过程就不说了,一路绿灯就行,安装完以后的状态以下图。
2.2Qt的项目建立
一直往下,中间过程就是本身改一下项目的名字和文件名字,直到出现下面的图,就是建立完成了。
上图是建立好的工程,这样就是能够运行的。
2.3 Qt的配置
这里说一下mingw版本的配置,此外msvc版本的Qt的编译器和调试器都是msvc(个人具体版本是MSVC2013)。
个人是32bit,而后全部的编译器和调试器都是套件自带的,不用另行下载。
好了,Qt的开始工做作好了,就能够开始学习了,就先到这里。
#include "widget.h"
#include <QApplication>
#include <QLabel>
#include <Qstring>
#include <QDebug>
#include <cstdio>
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
// QWidget w;
// QLabel label("hello world");
// w.show();
// label.show();
// QString str1,str2;
// str1="hello microgrid power ";
// str2="this is microgrid power ";
// str1+=str2;
// str1.append("welcome to microgrid power");
// str1=QString("dcdc双向变换器 %1 %2").arg(str1).arg(str2);
// QLabel *label=new QLabel(str1);
// label->show();
// QList<QString> list;
// {
// QString str("this is a test string");
// list<<str;
// list<<str;
// }
// qDebug()<<list[0]<<list[1];
QList<int> list;
for(int j=0;j<10;j++)
list.insert(list.end(),j);
QList<int>::iterator i;
for(i=list.begin();i!=list.end();i++)
{
qDebug()<<*i<<" ";
}
return a.exec();
}
1 #include <QCoreApplication>
2 #include <QString>
3 #include <QtDebug>
4 int main(int argc, char *argv[])
5 {
6 QCoreApplication a(argc, argv);
7 QMap<QString,QString>map;
8 map.insert("lili","1990");
9 map.insert("wangli","1992");
10 map.insert("zhangli","1989");
11 QMap<QString,QString>::const_iterator i;
12 for(i=map.constBegin();i!=map.constEnd();i++)
13 qDebug()<<" "<<i.key()<<" "<<i.value();
14 QMap<QString,QString>::iterator mi;
15 mi=map.find("lili");
16 if(mi!=map.end())
17 mi.value()="1995";
18 QMap<QString,QString>::const_iterator modi;
19 for(modi=map.constBegin();modi!=map.constEnd();modi++)
20 qDebug()<<" "<<modi.key()<<" "<<modi.value();
21 return a.exec();
22 }