一、背景
不得不说Qt是一个很强大的类库,不管是做项目还是做产品,Qt自身封装的东西就已经非常全面了,我们今天的这篇文章就是模拟了Qt读写ini文件的一个操作,当然是由于一些外力原因,我们决定自己来完善下这个功能。好的,那么现在就让我们隆重的请出今天的主角--QSettings。这个类能干嘛呢? 答案就是:读写注册表或者读写ini文件,这对于我们做应用程序时记录一些可持久化数据非常有用。
二、QSettings访问ini文件
QSettings访问ini文件相对来说比较简单,我们只需要构造一个QSettings对象,传入文件名称和文件存储格式即可,如图1所示。但同时QSettings也有一些局限,如下:
1、QSettings的编码问题(QTBUG15543、QTBUG19552)
2、QSettings的key不能为中文
3、当在一个嵌套作用域多次构造QSettings时并设置了编码,此时访问文件设置的编码会失效
由于QSettings有一些限制,也就引出了我们这篇文章的内容,使用xml模拟ini文件,下面我们主要分析下怎么使用xml文件模拟ini文件,需要的接口并不多,读、写、新增和删除。
图1 QSettings读写ini
三、xml文件读写
读写xml文件的方式有很多,Qt提供了2种比较常用的方式:DOM和SAX,详情可以参看:Qt学习之XML读写操作小结。初次之外C++还提供了几种库用于操作xml文件,比如:libxml2、tinyXml等。
此处我们模拟的是QSettings读写ini文件,因此使用Qt自带的DOM方式读写xml,操作起来相对容易。
如图2所示,是我们xml文件操作类的几个重要接口和成员,前边4个公有接口分别对应。读、写、新增和删除;成员变量m_filePath表示加载的xml文件路径,m_mItemMap表示2级的ini文件结构
图2 xml文件操作头文件
1、读xml文件
1 void xmlOperate::readXml( const QString & filePath )
2 {
3 if(filePath.isEmpty())
4 return;
5
6 QFile file(filePath);
7 if(file.open(QFile::ReadOnly | QFile::Text) == false)
8 return;
9
10 QDomDocument domDocument;
11 QString error;
12 int row = 0, column = 0;
13 if(domDocument.setContent(&file, false, &error, &row, &column) == false)
14 return file.close();
15
16 if(domDocument.isNull())
17 return file.close();
18
19 m_mItemMap.clear();
20
21 QDomElement rootElement = domDocument.documentElement();
22 QDomNodeList groupList = rootElement.childNodes();
23 for(int i = 0; i < groupList.count(); ++i)
24 {
25 QDomNode groupNode = groupList.item(i);
26 QDomElement groupElement = groupNode.toElement();
27 QString groupName = groupElement.attribute("name");
28
29 QMap<QString, QString> items;
30 QDomNodeList itemList = groupElement.childNodes();
31 for (int j = 0; j < itemList.count(); ++j)
32 {
33 QDomNode itemNode = itemList.item(j);
34 QDomElement itemElement = itemNode.toElement();
35 QString itemName = itemElement.attribute("name");
36 QString itemText = itemElement.text();
37
38 if (itemName.isEmpty())
39 {
40 items[QString::number(j)] = itemText;
41 }
42 else
43 {
44 items[itemName] = itemText;
45 }
46 }
47 m_mItemMap[groupName] = items;
48 }
49
50 file.close();
51
52 qDebug() << m_mItemMap;
53 }
2、保存xml文件
1 void xmlOperate::writeXml(const QString & filePath)
2 {
3 QDomDocument domDocument;
4 QString strHeader( "version=\"1.0\" encoding=\"UTF-8\"" );
5 domDocument.appendChild( domDocument.createProcessingInstruction("xml", strHeader) );
6
7 QDomElement root = domDocument.createElement("groups");
8 domDocument.appendChild(root);
9
10 for(auto iter = m_mItemMap.begin(); iter != m_mItemMap.end(); ++iter)
11 {
12 QDomElement groupNode = domDocument.createElement("group");
13 groupNode.setAttribute("name", iter.key());
14
15 QMap<QString, QString> items = iter.value();
16 for (auto iter2 = items.begin(); iter2 != items.end(); ++iter2)
17 {
18 QString key = iter2.key();
19 QString value = iter2.value();
20
21 QDomElement itemNode = domDocument.createElement("item");
22 itemNode.setAttribute("name", key);
23
24 QDomText textNode = domDocument.createTextNode(value);
25 itemNode.appendChild(textNode);
26
27 groupNode.appendChild(itemNode);
28 }
29 root.appendChild(groupNode);
30 }
31
32 QFile file(filePath);
33 if(file.open(QFile::WriteOnly | QFile::Text))
34 {
35 QTextStream out(&file);
36 domDocument.save(out, 4);
37 file.close();
38 }
39 qDebug() << m_mItemMap;
40 }
3、插入项
1 bool xmlOperate::addItem( const QString & value, const QString & group, const QString & key )
2 {
3 if (value.isEmpty() || group.isEmpty())
4 {
5 return false;
6 }
7
8 if (key.isEmpty())
9 {
10 int count = m_mItemMap[group].size();
11 m_mItemMap[group][QString::number(count)] = value;
12 }
13 else
14 {
15 m_mItemMap[group][key] = value;
16 }
17
18 writeXml(m_filePath);
19
20 return true;
21 }
4、删除项
1 bool xmlOperate::removeItem( const QString & value, const QString & group, const QString & key /*= ""*/ )
2 {
3 if (value.isEmpty() || group.isEmpty())
4 {
5 return false;
6 }
7
8 if (key.isEmpty())
9 {
10 int count = m_mItemMap[group].size();
11 if (count == 0)
12 {
13 return false;
14 }
15
16 auto iter = m_mItemMap[group].begin();
17 while (iter != m_mItemMap[group].end())
18 {
19 if (iter.value() == value)
20 {
21 iter = m_mItemMap[group].erase(iter);
22 }
23 else
24 {
25 ++iter;
26 }
27 }
28 }
29 else
30 {
31 m_mItemMap[group].remove(key);
32 }
33
34 writeXml(m_filePath);
35
36 return true;
37 }
四、示例程序下载
如图3是测试代码生成的测试结果,group相当于ini文件中的一个分组,item表示分组中的一项
图3 测试程序结果
来源:oschina
链接:https://my.oschina.net/u/4313343/blog/3906306