C:细分错误,为什么以及如何解决

时间:2019-12-08 16:58:14

标签: c libxml2


我目前正在学习编程方法,并且可以确定我的代码中充满了错误。
执行此代码时出现段错误,我也不知道为什么。

#include <stdio.h>
#include <libxml/parser.h>
#include <string.h>
#include <stdlib.h>


typedef struct
{
        int ac; /**< The player armor class. */
        char *class; /**< The player class. */
        char *cname; /**< The player character name. */
        int cp; /**< The player money (copper pieces, 100 CP = 1 GP). */
        int gp; /**< The player money (gold pieces). */
        int hp; /**< The player hit points. */
        char *name; /**< The player name. */
        int sp; /**< The player money (silver pieces, 10 SP = 1 GP). */
} player_t;

void parsePlayer(xmlDocPtr doc, xmlNodePtr node){
    int i = 1;
    player_t players[3];
    xmlNodePtr cur;

    node = node->xmlChildrenNode;

    while(node!=NULL){
        if (!xmlStrcmp(node->name, (const xmlChar *) "player"))
        {
            players[i].name = (char *) xmlGetProp(node, (const xmlChar *)"name");
            printf("player %d name: %s\n", i, players[i].name);
            cur = node->xmlChildrenNode;
            players[i].ac = strtol((char *) xmlNodeListGetString(doc, cur->xmlChildrenNode, 1),NULL,10);
            i++;
        }

        node = node->next;
    }
}

void parsePlayers(xmlDocPtr doc){
    xmlNodePtr node;

    node = xmlDocGetRootElement(doc);

    node = node->xmlChildrenNode; 

    while(node!=NULL){
        if (!xmlStrcmp(node->name, (const xmlChar *) "players")){
            parsePlayer(doc, node);
        }
        node = node->next;
    }
}

int main(int argc, char const *argv[]) {

    xmlDocPtr doc = xmlParseFile(argv[1]);

    if (doc != NULL) printf("DEBUG: File parsed.\n");   
    else { 
        fprintf(stderr,"Unable to parse the document \n");
        return 1;
    }

    parsePlayers(doc);
    return 0;
}

代码显示如下:

DEBUG: File parsed.
player 1 name: Camille /* correct */
Segmentation Fault

我已经尝试寻找答案,但似乎我对理解它们的能力还不够高:/。
这是针对学校的一个项目(我必须填充几个player_t结构)。
我认为错误不是来自XML,因此无需在此处发布。
我已经在这个错误上停留了几天了,我很想了解如何纠正它。
感谢您的阅读:)。

编辑:感谢您的回答,gdb说:

Program received signal SIGSEGV, Segmentation fault.
__GI_____strtol_l_internal (nptr=0x0, endptr=0x0, base=10, group=<optimized out>, 
    loc=0x7ffff7a0f560 <_nl_global_locale>) at ../stdlib/strtol_l.c:292
292 ../stdlib/strtol_l.c: no such file or directory. 

我不知道这是什么意思:(。

编辑2:好吧,错误确实来自xml,这里是:

<ddg name="The frozen cave">
        <date>
                <day>01</day>
                <month>09</month>
                <year>2019</year>
        </date>
        <dmname>Pierre</dmname>
        <players>
                <player name="Camille">
                        <ac>5</ac>
                        <class>Thief</class>
                        <cname>Camigolas</cname>
                        <hp>4</hp>
                        <money>
                                <cp>10</cp>
                                <gp>10</gp>
                                <sp>10</sp>
                        </money>
                </player>
                <player name="Jean">
                        <ac>4</ac>
                        <class>Fighter</class>
                        <cname>Jeragorn</cname>
                        <hp>8</hp>
                        <money>
                                <cp>3</cp>
                                <gp>1</gp>
                                <sp>2</sp>
                        </money>
                </player>
                <player name="Paul">
                        <ac>5</ac>
                        <class>Cleric</class>
                        <cname>Paumli</cname>
                        <hp>6</hp>
                        <money>
                                <cp>9</cp>
                                <gp>5</gp>
                                <sp>9</sp>
                        </money>
                </player>
        </players>
</ddg>

我尝试了几件事,最终发现播放器的xmlChildrenNode是文本。
我猜我只需要循环,直到找到想要的元素,但是没有人知道为什么它返回文本吗? (我只是printf cur-> name来发现这一点)。
谢谢大家的回答:)!

2 个答案:

答案 0 :(得分:1)

nptr=0x0可能意味着使用参数strtol设置为strtol来调用nptr(或更准确地说是0x0所调用的函数)。参数nptr最希望指向有效的字符串,但没有。

这可能意味着xmlNodeListGetString()返回零或NULL,具体取决于您更喜欢如何调用该值。

解析外部文件时,您应该始终对文件内容进行错误检查。

我猜想cur->xmlChildrenNode并没有您期望的价值,但是如果没有更多信息,很难说。

顺便说一句,c是具有从零开始的数组的那些语言之一,因此您应该有int i = 0。 (基于一个的数组是 evil 。)通过在players[0]处使用一个虚拟元素来模拟基于一个的数组,不要成为异端。

答案 1 :(得分:1)

  

我尝试了几件事,最终发现播放器的xmlChildrenNode是文本。我想我只需要循环运行,直到找到想要的元素,但是没有人知道为什么它返回文本

XML文件中元素之间的所有空格都很重要。 节点将有11个节点子节点:

  • 一个文本节点,在之后紧接换行符,在
  • 之前紧跟空格
  • 一个元素节点,即(它将有一个文本节点子元素,即ac值)
  • 文本节点,在之后的换行符,在
  • 之前的空格
  • 元素节点,即
  • 等...

因此,您在下看到的第一个“文本”子级是换行符和之前的空格。

您可能希望将XML_PARSE_NOBLANKS传递到xmlParseFile中,这告诉解析器像这样跳过“中间”空白。然后将具有子项,然后具有子项,依此类推,而中间没有文本节点。