0

我一直在寻找这个问题的答案,但是我在任何地方找到的所有 SAX 资源都比我希望的要少一些。我正在为一家餐厅编写一个 android 应用程序,它可以让客人通过应用程序访问很长的列表,而不是翻阅一本书。我的 xml 看起来像这样:

<bar>
    <liquor>
        <type>American Rye</type>
        <distillery>Sazerac<distillery>
        <bottling>18 Year</bottling>
        <place>Frankfort, KY</place>
        <proof>90</proof>
        <price>20<price>
    </liquor>

    <beer>
        <type>American Microbrew</type>
        <brewery>New Belgium</brewery>
        <bottling>La Folie Sour Brown 750ml</bottling>
        <place>Fort Collins, CO</place>
        <price>20</price>
    </beer>
</bar>

当我只有几百杯酒时,它运作良好。但是,因为我两次使用某些元素名称,例如“类型”和“价格”,所以事情变得一团糟。这是我的解析器:

public class BeerParser extends DefaultHandler {

private ArrayList<Beer> BeerL;
private boolean pastTheLiquor = false;

public ArrayList<Beer> getItems(String ArrayType){
    ArrayList<Beer> tmpItem = new ArrayList<Beer>();
    for (Beer beer : BeerL){
        if (beer.getType().equals(ArrayType)){
        tmpItem.add(beer);
            }
        }
    return tmpItem;
}


InputStream barXmlInputStream;

String tmpValue;


Beer beerTmp;



public BeerParser(InputStream barXmlInputStream) {



    this.barXmlInputStream = barXmlInputStream;

    BeerL = new ArrayList<Beer>();

    parseDocument();

    printDatas();

}

private void parseDocument() {



    SAXParserFactory factory = SAXParserFactory.newInstance();

    try {

        SAXParser parser = factory.newSAXParser();

        parser.parse(barXmlInputStream, this);



    } catch (ParserConfigurationException e) {

        System.out.println("ParserConfig error");

    } catch (SAXException e) {

        System.out.println("SAXException : xml not well formed");

    } catch (IOException e) {

        System.out.println("IO error");

    }

}

private void printDatas() {


    for (Beer tmpB : BeerL) {

        System.out.println(tmpB.toString());

    }


}

@Override

public void startElement(String s, String s1, String elementName, Attributes attributes) throws SAXException {


    if (elementName.equalsIgnoreCase("beer")) {

        pastTheLiquor = true;
        beerTmp = new Beer();



    }

}

@Override

public void endElement(String s, String s1, String element) throws SAXException {


    if (element.equals("beer")) {

            BeerL.add(beerTmp);

        }
    if (pastTheLiquor){
        if (element.equalsIgnoreCase("type")) {

            beerTmp.setType(tmpValue);

        }
        if (element.equalsIgnoreCase("brewery")) {

            beerTmp.setBrewery(tmpValue);

        }
        if (element.equalsIgnoreCase("bottling")) {

            beerTmp.setBottling(tmpValue);
            beerTmp.hasBottling = true;

        }

        if (element.equalsIgnoreCase("price")) {

            beerTmp.setPrice(tmpValue);

        }

        if (element.equalsIgnoreCase("place")) {

            beerTmp.setPlace(tmpValue);

        }
    }

}



@Override

public void characters(char[] ac, int i, int j) throws SAXException {

    tmpValue = new String(ac, i, j);

}

}

因此,酒先于啤酒出现,因此,由于解析器在看到“啤酒”之前先看到“类型”,它试图调用从未实例化的 Beer 对象 beerTmp 的“setType()”函数。我尝试使用布尔值,它会等到解析器看到“啤酒”的第一个实例,但我得到一个空列表,这真的让我很沮丧,因为制作一系列酒的几乎相同的解析器正在工作壮观。

有没有一种简单的方法可以跳过文件中的酒?我在布尔值的正确轨道上吗?我应该把 SAX 解析器扔出窗外并使用其他东西吗?谢谢你。

4

2 回答 2

0

sax 解析器为所有元素调用回调,如果您不想考虑其中的一些,只需快速执行“return;”。在解析之前无法将它们过滤掉。

有没有一种简单的方法可以跳过文件中的酒?我在布尔值的正确轨道上吗?

您可以使用堆栈结构来存放稍后需要考虑的数据(推送)(弹出)。

SAX 绝对比其他 XML API 更难驯服,因为在某些情况下,它是唯一的选择。所以,不要放弃它;之前或之后你会需要它。

于 2013-01-09T19:11:42.377 回答
0

以下简化的视觉作品。注意

1)你的xml有错误,例如 <type>American Rye</liquor>

2) equalsIgnoreCase 无关紧要,因为 xml 区分大小写。

3) JAXB 最适合这项工作

public class BeerParser extends DefaultHandler {
    private List<Beer> list = new ArrayList<>();
    private Beer beer;
    private String element;

    public static void main(String[] args) throws Exception {
        SAXParserFactory factory = SAXParserFactory.newInstance();
        SAXParser parser = factory.newSAXParser();
        BeerParser bp = new BeerParser();
        parser.parse(new File("1.xml"), bp);
        System.out.println(bp.list);
    }

    @Override
    public void startElement(String s, String s1, String element, Attributes attributes) {
        if (element.equals("beer")) {
            beer = new Beer();
        }
        this.element = element;
    }

    @Override
    public void endElement(String s, String s1, String element) {
        if (element.equals("beer")) {
            list.add(beer);
            beer = null;
        }
    }

    @Override
    public void characters(char[] ac, int i, int j) {
        if (beer != null) {
            String value = new String(ac, i, j);
            if (element.equals("type")) {
                beer.setType(value);
            } else if (element.equals("brewery")) {
                beer.setBrewery(value);
            } else if (element.equals("bottling")) {
                beer.setBottling(value);
            } else if (element.equals("price")) {
                beer.setPrice(value);
            } else if (element.equals("place")) {
                beer.setPlace(value);
            }
        }
    }
}
于 2013-01-09T19:32:51.667 回答