使用libxml2进行递归XPath查询的最有效方法是什么?

时间:2009-11-05 19:50:15

标签: c++ xpath libxml2

我为libxml2编写了一个C ++包装器函数,这使我可以轻松地对XML文档进行查询:

bool XPathQuery(
    const std::string& doc,
    const std::string& query,
    XPathResults& results); 

但是我遇到了一个问题:我需要能够对第一次查询的结果进行另一次XPath查询。

目前我通过将整个子文档存储在我的XPathResult对象中来实现这一点,然后将XPathResult.subdoc传递给XPathQuery函数。这非常低效。

所以我想知道... libxml2是否提供了任何可以轻松存储xpath查询的上下文(可能是对节点的引用?)的内容,然后使用该引用作为xpath根执行另一个查询?

1 个答案:

答案 0 :(得分:5)

您应该重复使用xmlXPathContext,只需更改其node成员。

#include <stdio.h>
#include <libxml/xpath.h>
#include <libxml/xmlerror.h>

static xmlChar buffer[] = 
"<?xml version=\"1.0\"?>\n<foo><bar><baz/></bar></foo>\n";

int
main()
{
  const char *expr = "/foo";

  xmlDocPtr document = xmlReadDoc(buffer,NULL,NULL,XML_PARSE_COMPACT);
  xmlXPathContextPtr ctx = xmlXPathNewContext(document);
  //ctx->node = xmlDocGetRootElement(document);

  xmlXPathCompExprPtr p = xmlXPathCtxtCompile(ctx, (xmlChar *)expr);
  xmlXPathObjectPtr res = xmlXPathCompiledEval(p, ctx);

  if (XPATH_NODESET != res->type)
    return 1;

  fprintf(stderr, "Got object from first query:\n");
  xmlXPathDebugDumpObject(stdout, res, 0);
  xmlNodeSetPtr ns = res->nodesetval;
  if (!ns->nodeNr)
    return 1;
  ctx->node = ns->nodeTab[0];
  xmlXPathFreeObject(res);

  expr = "bar/baz";
  p = xmlXPathCtxtCompile(ctx, (xmlChar *)expr);
  res = xmlXPathCompiledEval(p, ctx);

  if (XPATH_NODESET != res->type)
    return 1;
  ns = res->nodesetval;
  if (!ns->nodeNr)
    return 1;
  fprintf(stderr, "Got object from second query:\n");
  xmlXPathDebugDumpObject(stdout, res, 0);

  xmlXPathFreeContext(ctx);
  return 0;
}