libxml问题获取标签的属性

时间:2019-05-10 07:47:05

标签: c xml-parsing libxml2

我只想获取属性对象的值,但是如果由于第一个节点未进入循环而无法在第一个节点之后工作,为什么?

这是我的xml文件:

     <msg><tag date="1557417027960" session="1697"><decision object="BAST04HEF" reliability="95" context="RO" x="796" y="371" 
 width="89" height="18" 
 direction="front"><jpeg></jpeg></decision></tag></msg>

这是我的代码:

int main(int argc, char **argv) {

  char         *docname;
  xmlDocPtr    doc;
  xmlNodePtr   cur;
  xmlChar      *object;

  if (argc < 2) {
    printf("Commande: %s nom_du_fichier\n", argv[0]);
    return EXIT_FAILURE;
  }


  docname = argv[1];

  doc = xmlParseFile(docname);
  cur = xmlDocGetRootElement(doc);

  cur = cur->xmlChildrenNode;
  while (cur != NULL) {
      if ((!xmlStrcmp(cur->name, (const xmlChar *)"decision"))) {
        object = xmlGetProp(cur, "object");
        printf("object: %s\n", object);
        xmlFree(object);
      }
      cur = cur->next;
  }
  xmlFreeDoc(doc);
  return EXIT_SUCCESS;
}

1 个答案:

答案 0 :(得分:2)

您的代码仅处理一级子级,即根元素的直接子级。

cur = xmlDocGetRootElement(doc);获取根元素。

cur = cur->xmlChildrenNode;获取根元素的第一个(直接)子元素。

在循环中,您获得了这个第一个孩子的所有兄弟姐妹,cur = cur->next;,但是您不处理他们可能的孩子。

您的XML代码段显示您至少具有三层:msg-tag-decision

如果您想处理所有decision元素,无论其父级是什么,都可以使用递归函数。

static void processChildren(xmlNodePtr cur)

int main(int argc, char **argv) {

  char         *docname;
  xmlDocPtr    doc;
  xmlNodePtr   cur;
  xmlChar      *object;

  if (argc < 2) {
    printf("Commande: %s nom_du_fichier\n", argv[0]);
    return EXIT_FAILURE;
  }


  docname = argv[1];

  doc = xmlParseFile(docname);
  cur = xmlDocGetRootElement(doc);

  processChildren(cur->xmlChildrenNode);

  xmlFreeDoc(doc);
  return EXIT_SUCCESS;
}

static void processChildren(xmlNodePtr cur)
{
    while (cur != NULL) {
        if ((!xmlStrcmp(cur->name, (const xmlChar *)"decision"))) {
            object = xmlGetProp(cur, "object");
            printf("object: %s\n", object);
            xmlFree(object);
        }
        else
        {
            processChildren(cur->children);
        }
        cur = cur->next;
    }
}