如何在QT中运行XPath查询?

时间:2009-08-17 08:49:00

标签: c++ qt xpath

如何在QT中运行XPath查询?

我需要在特定属性中对具有特定值的某些标记进行排序。 QXmlQuery文档是清晰易读的。

我正在解析的模式是Rhythmbox DB格式:


<rhythmdb version="1.6">
  <entry type="ignore">
    <title></title>
    <genre></genre>
    <artist></artist>
    <album></album>
    <location>file:///mnt/disk/music/Cover.jpg</location>
    <mountpoint>file:///mnt/disk</mountpoint>
    <mtime>1222396828</mtime>
    <date>0</date>
    <mimetype>application/octet-stream</mimetype>
    <mb-trackid></mb-trackid>
    <mb-artistid></mb-artistid>
    <mb-albumid></mb-albumid>
    <mb-albumartistid></mb-albumartistid>
    <mb-artistsortname></mb-artistsortname>
  </entry>
  <entry type="song">
    <title>Bar</title>
    <genre>Foobared Music</genre>
    <artist>Foo</artist>
    <album>The Great big Bar</album>
    <track-number>1</track-number>
    <disc-number>1</disc-number>
    <duration>208</duration>
    <file-size>8694159</file-size>
    <location>file:///media/disk/music/01-Foo_-_Bar.ogg
    <mountpoint>file:///media/disk
    <mtime>1216995840</mtime>
    <first-seen>1250478814</first-seen>
    <last-seen>1250478814</last-seen>
    <bitrate>301</bitrate>
    <date>732677</date>
    <mimetype>application/x-id3</mimetype>
    <mb-trackid></mb-trackid>
    <mb-artistid></mb-artistid>
    <mb-albumid></mb-albumid>
    <mb-albumartistid></mb-albumartistid>
    <mb-artistsortname></mb-artistsortname>
  </entry>
</rhythmdb>

这是您的基本XML Schema,它包含一组结构化条目。我的目的是过滤掉“忽略”类型的条目。

2 个答案:

答案 0 :(得分:11)

相关文件位于:http://qt-project.org/doc/qt-4.8/qxmlquery.html#running-xpath-expressions

我遇到的解决方案是使用QXmlQuery生成XML文件,然后使用QDomDocument再次解析它。


RhythmboxTrackModel::RhythmboxTrackModel()
{
    QXmlQuery query;
    QXmlQuery entries;
    QString res;
    QDomDocument rhythmdb;


    /*
     * Try and open the Rhythmbox DB. An API call which tells us where
     * the file is would be nice.
     */
    QFile db(QDir::homePath() + "/.gnome2/rhythmbox/rhythmdb.xml");
    if ( ! db.exists()) {
        db.setFileName(QDir::homePath() + "/.local/share/rhythmbox/rhythmdb.xml");
        if ( ! db.exists())
            return;
    }

    if (!db.open(QIODevice::ReadOnly | QIODevice::Text))
        return;

    /*
     * Use QXmlQuery to execute and XPath query. Check the version to
     * make sure.
     */
    query.setFocus(&db);
    query.setQuery("rhythmdb[@version='1.6']/entry[@type='song']");
    if ( ! query.isValid())
        return;

    query.evaluateTo(&res);
    db.close();


    /*
     * Parse the result as an XML file. These shennanigans actually
     * reduce the load time from a minute to a matter of seconds.
     */
    rhythmdb.setContent("" + res + "");
    m_entryNodes = rhythmdb.elementsByTagName("entry");


    for (int i = 0; i < m_entryNodes.count(); i++) {
        QDomNode n = m_entryNodes.at(i);
        QString location = n.firstChildElement("location").text();

        m_mTracksByLocation[location] = n;
    }

    qDebug() << rhythmdb.doctype().name();
    qDebug() << "RhythmboxTrackModel: m_entryNodes size is" << m_entryNodes.size();
}

如果有人想知道,这是我的代码来自Mixxx project的最新分支,特别是features_looping分支。

我不喜欢这个解决方案的事情是:

  • 解析XML两次
  • 将结果与开始和结束标记连接。

答案 1 :(得分:-2)

如果它符合您的解析要求,您可以使用基于SAX的阅读器而不是基于DOM的阅读器。将QXmlSimpleReader与子类QXmlDefaultHandler一起使用,您可以在扫描文档时访问XPath查询的每个元素及其属性。我认为这种方法比基于DOM的方法更快;你不必阅读任何东西两次,它已经内置到Qt中。这里有一个例子:“使用SAX阅读”下的http://www.digitalfanatics.org/projects/qt_tutorial/chapter09.html