我們知道對XML的操作有兩種方法,即DOM方式和SAX方式。二者主要區(qū)別是:DOM實(shí)現(xiàn)方式操作非常簡單,但不適合處理過大文件;而SAX實(shí)現(xiàn)方式是能處理很大的XML文件,但是需要開發(fā)者寫一些復(fù)雜的代碼。Qt提供了對應(yīng)于這兩種用于讀取、操作和編寫XML的實(shí)現(xiàn)類,分別是QDomDocument類和QXmlStreamReader類,由于在項(xiàng)目中涉及的文件不大,因此我們選用QDomDocument類來處理。
項(xiàng)目中涉及便簽的增刪改查,對應(yīng)于XML文件中相應(yīng)標(biāo)記的讀、寫和修改,下面分別介紹:
1. 創(chuàng)建節(jié)點(diǎn),將其寫入XML文件,主要操作包括:
1).創(chuàng)建根節(jié)點(diǎn):QDomElement root = doc.documentElement("rootName " );
2).創(chuàng)建元素節(jié)點(diǎn):QDomElement element =?doc.createElement("nodeName");
3).添加元素節(jié)點(diǎn)到根節(jié)點(diǎn):root. appendChild(element);
4).創(chuàng)建元素文本:QDomText nodeText=doc.createTextNode("text");
5).添加元素文本到元素節(jié)點(diǎn):element. appendChild(nodeText);
在本項(xiàng)目中,假設(shè)便簽的屬性有序號、名字、內(nèi)容、字體、字號、顏色、粗細(xì)、斜體、下劃線這幾項(xiàng),則在文件中添加一個(gè)便簽節(jié)點(diǎn)的操作如下:
QDomDocument doc;
instruction = doc.createProcessingInstruction("xml","version="1.0" encoding="UTF-8"");
doc.appendChild(instruction);
QDomElement root = doc.createElement("Notes");
doc.appendChild(root);
QDomElement note = doc.createElement("note");
root.appendChild(note);
QDomElement no = doc.createElement("no");
note.appendChild(no);
...
...
QDomText no_text = doc.createTextNode("001");
...
...
則得到一個(gè)便簽節(jié)點(diǎn),將其保存到test.xml文件中,代碼如下:
QFile file("test.xml");
if (!file.open(QIODevice::WriteOnly | QIODevice::Truncate |QIODevice::Text))
???? return ;
QTextStream out(&file);
out.setCodec("UTF-8");
doc.save(out,4,QDomNode::EncodingFromTextStream);
file.close();
則test.xml文件:
???
???????
???????
???????
??????? Script MT Bold
???????
???????
???????
???????
???????
???
上面是創(chuàng)建一個(gè)便簽節(jié)點(diǎn),若要繼續(xù)添加便簽節(jié)點(diǎn),則需要在已有節(jié)點(diǎn)后增加一個(gè)新節(jié)點(diǎn),并重寫入XML文件。
2. 加載、查找便簽時(shí)要讀取XML文檔中的節(jié)點(diǎn)信息,DOM實(shí)現(xiàn)方式是將整個(gè)文檔當(dāng)作一個(gè)對象來裝入內(nèi)存進(jìn)行處理,然后開發(fā)者可以訪問這個(gè)對象中的每一個(gè)節(jié)點(diǎn),每一個(gè)節(jié)點(diǎn)對應(yīng)XML文件里的一個(gè)標(biāo)記。
?????? 主要操作包括:
?????? 1).讀取根節(jié)點(diǎn):QDomElement root = doc.documentElement();
2).讀取第一個(gè)子節(jié)點(diǎn):QDomNode node = root.firstChild();
3).讀取下一個(gè)子節(jié)點(diǎn):node = node.nextSibling();
4).匹配結(jié)點(diǎn)標(biāo)記:node.toElement().tagName() == "note"
5).讀取節(jié)點(diǎn)文本:no = childNode.toText().data();
以下是項(xiàng)目中讀取便簽屬性的函數(shù)實(shí)現(xiàn)代碼:
void MainWindow::parseAttr(const QDomElement &element)
{
??? QString no,name,content,font,fontSize,color;
??? QDomNode node = element.firstChild();
??? while (!node.isNull()) {
??????? if (node.toElement().tagName() == "note") {//匹配note節(jié)點(diǎn)
??????????? parseAttr(node.toElement());
??????? } else if (node.toElement().tagName() == "no") {//匹配屬性no
??????????? QDomNode childNode = node.firstChild();
??????????????? if (childNode.nodeType() == QDomNode::TextNode) {
???????????????????? no = childNode.toText().data();
??????????????? }
??????? }
??????? else if (node.toElement().tagName() == "name") //匹配屬性name
????????????????? ...
????????????????? ...
??????? node = node.nextSibling();//讀取兄弟節(jié)點(diǎn)
??? }
}
3. 刪除便簽時(shí),要?jiǎng)h除相應(yīng)的XML節(jié)點(diǎn),用到的主要函數(shù)為:root.removeChild(node); 但在刪除某個(gè)節(jié)點(diǎn)后要重寫整個(gè)文件。
以上對XML文件的重寫操作是必須的,因此在文件的打開方式中要加上QIODevice::Truncate,表示覆蓋重寫。目前還沒有找到可以直接修改文件的方法,但若文件很大的情況下,必須考慮相應(yīng)的效率問題。
由于本項(xiàng)目在啟動(dòng)時(shí)需要將所有便簽加載到內(nèi)存,因此選用DOM方式比較合適,但如果處理的XML文件很大,而且不需要全部讀到內(nèi)存時(shí),可以用SAX實(shí)現(xiàn)方式,它按階段將文檔讀取到內(nèi)存中,在碰到標(biāo)簽或者其它階段的時(shí)候,可以調(diào)用開發(fā)者預(yù)先設(shè)計(jì)好的回調(diào)函數(shù)去處理,這樣效率比DOM方式更高。