3

次のxmlファイルがあります。

<?xml version="1.0"?>
<CONFIG>
    <FUNCTION>
        <NAME>FUNCT0</NAME>
        <CALLS>
            <FUNCTION>
            <NAME>FUNCT0_0</NAME>
            </FUNCTION>
        </CALLS>
        <CALLS>
            <FUNCTION>
            <NAME>FUNCT0_1</NAME>
            </FUNCTION>
        </CALLS>
    </FUNCTION>
    <FUNCTION>
        <NAME>FUNCT1</NAME>
    </FUNCTION>
</CONFIG>

関数の名前と、関数が呼び出すサブ関数を含む ArrayList の両方を格納する FunctionInfo というクラスがあります。

オブジェクト内にサブ関数を再帰的に格納する最上位関数を含む ArrayList を作成したいと考えています。

再帰の無限の深さで機能するには、これが必要です。

私の質問は、このタスクを実行できる再帰的 XML パーサーを作成する最も簡単な方法は何ですか?

編集:私はJavaで働いています。

ありがとう :)

4

2 に答える 2

8

ファイルが巨大でない限り、Java DOM パーサーを使用できます (DOM パーサーはファイルをメモリに保持します)。

ノード (ルートから開始) を指定すると、その子を列挙し、各子に対して同じ関数を再帰的に呼び出すことができます。

import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;

import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;

public class RecursiveDOM {
    public static void main(final String[] args) throws SAXException, IOException, ParserConfigurationException {
        new RecursiveDOM("file.xml");
    }

    public RecursiveDOM(final String file) throws SAXException, IOException, ParserConfigurationException {
        final DocumentBuilderFactory dbfac = DocumentBuilderFactory.newInstance();
        final DocumentBuilder docBuilder = dbfac.newDocumentBuilder();
        final Document doc = docBuilder.parse(this.getClass().getResourceAsStream(file));
        final List<String> l = new ArrayList<String>();
        parse(doc, l, doc.getDocumentElement());
        System.out.println(l);
    }

    private void parse(final Document doc, final List<String> list, final Element e) {
        final NodeList children = e.getChildNodes();
        for (int i = 0; i < children.getLength(); i++) {
            final Node n = children.item(i);
            if (n.getNodeType() == Node.ELEMENT_NODE) {
                list.add(n.getNodeName());
                parse(doc, list, (Element) n);
            }
        }
    }

}

結果:

[FUNCTION, NAME, CALLS, FUNCTION, NAME, CALLS, FUNCTION, NAME, FUNCTION, NAME]
于 2012-11-08T19:09:35.187 に答える
1

また、JAXB (XML へのクラスのマッピング) の使用を検討することもできます。これにより、Java クラスのツリーが生成されます。適切に実装されている場合、各クラスは特定のタイプの関数を実装できるため、ツリーを再帰的に「計算」できます。

これは基本的な JAXB 実装です。Config.unmarshal を使用して、InputStream から再構築します。

import java.io.OutputStream;
import java.io.Reader;

import javax.xml.bind.JAXBContext;
import javax.xml.bind.JAXBException;
import javax.xml.bind.Marshaller;
import javax.xml.bind.Unmarshaller;
import javax.xml.bind.annotation.XmlElement;
import javax.xml.bind.annotation.XmlRootElement;

public class TestJAXB {

    public static void main(final String[] args) throws JAXBException {
        final Function e5 = new Function("N5", null);
        final Function e6 = new Function("N6", null);
        final Function e4 = new Function("N4", null);
        final Function e2 = new Function("N2", ((new Call[] { new Call(e4) })));
        final Function e3 = new Function("N3", ((new Call[] { new Call(e5) })));
        final Function e1 = new Function("N1", ((new Call[] { new Call(e2), new Call(e3) })));
        new Config(new Function[] { e1, e6 }).marshall(System.out);
    }

    @XmlRootElement(name = "CONFIG")
    static class Config {
        @XmlElement(name = "FUNCTION")
        private Function[] functions;

        public Config(final Function[] f) {
            this.functions = f;
        }

        protected Config() {
        }

        public void marshall(final OutputStream out) throws JAXBException {
            final JAXBContext jaxbContext = JAXBContext.newInstance(this.getClass());
            final Marshaller marshaller = jaxbContext.createMarshaller();
            marshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, Boolean.TRUE);
            marshaller.marshal(this, out);
        }

        public final static Config unmarshall(final Reader r) throws JAXBException {
            final JAXBContext jaxbContext = JAXBContext.newInstance(Config.class);
            final Unmarshaller unmarshaller = jaxbContext.createUnmarshaller();
            return (Config) unmarshaller.unmarshal(r);
        }

    }

    @XmlRootElement(name = "FUNCTION")
    static class Function {
        @XmlElement(name = "NAME")
        String name;

        @XmlElement(name = "CALLS")
        Call[] calls;

        public Function(final String name, final Call[] calls) {
            this.name = name;
            this.calls = calls;
        }

        protected Function() {
        }

    }

    @XmlRootElement(name = "CALL")
    static class Call {
        @XmlElement(name = "FUNCTION")
        Function f;

        protected Call() {
        }

        public Call(final Function f) {
            this.f = f;
        }

    }
}
于 2012-11-08T20:20:56.783 に答える