我为libxml2编写了一个C ++包装器函数,这使我可以轻松地对XML文档进行查询:
bool XPathQuery(
const std::string& doc,
const std::string& query,
XPathResults& results);
但是我遇到了一个问题:我需要能够对第一次查询的结果进行另一次XPath查询。
目前我通过将整个子文档存储在我的XPathResult对象中来实现这一点,然后将XPathResult.subdoc传递给XPathQuery函数。这非常低效。
所以我想知道... libxml2是否提供了任何可以轻松存储xpath查询的上下文(可能是对节点的引用?)的内容,然后使用该引用作为xpath根执行另一个查询?
答案 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;
}