DocumentTraversal の NodeIterator を使用して、xml ドキュメント全体をトラバースできることは誰もが知っています。私のアプリケーションには追加の作業が必要なので、Java Stack<> をサポートする独自の XML トラバーサーを作成することにしました。
これが私のコードです(私はコーディングが苦手なので、コードとロジックが乱雑に見えるかもしれません)。
public class test
{
    private static Stack<Node> gStack = new Stack<Node>();
    public static void main(String[] args) throws XPathExpressionException
    {
        String str =
            "<section>"
                + "<paragraph>This example combines regular wysiwyg editing of a document with very controlled editing of semantic rich content. The main content can be"
                + "edited like you would in a normal word processor. Though the difference is that the content remains schema valid XML because Xopus will not allow you to perform actions"
                + "on the document that would render it invalid.</paragraph>"
                + "<paragraph>The table is an example of controlled style. The style of the table is controlled by three attributes:</paragraph>"
                + "<unorderedlist>"
                + "<item><paragraph><emphasis>alternaterowcolor</emphasis>, do all rows have the same color, or should the background color alternate?</paragraph></item>"
                + "<item><paragraph><emphasis>border</emphasis>, a limited choice of border styles.</paragraph></item>"
                + "<item><paragraph><emphasis>color</emphasis>, a limited choice of colors.</paragraph></item>"
                + "</unorderedlist>"
                + "<paragraph>You have quite some freedom to style the table, but you can't break the predefined style.</paragraph>"
                + "</section>";
        Document domDoc = null;
        try
        {
            DocumentBuilderFactory docFactory = DocumentBuilderFactory.newInstance();
            DocumentBuilder docBuilder = docFactory.newDocumentBuilder();
            ByteArrayInputStream bis = new ByteArrayInputStream(str.getBytes());
            domDoc = docBuilder.parse(bis);
        }
        catch (Exception e)
        {
            e.printStackTrace();
        }
        Element root = null;
        NodeList list = domDoc.getChildNodes();
        for (int i = 0; i < list.getLength(); i++)
        {
            if (list.item(i) instanceof Element)
            {
                root = (Element) list.item(i);
                break;
            }
        }
        NodeList nlist = root.getChildNodes();
        System.out.println("root = " + root.getNodeName() + "  child count = " + nlist.getLength());
        domTraversor(root);
    }
    private static void domTraversor(Node node)
    {
        if (node.getNodeName().equals("#text"))
        {
            System.out.println("textElem = " + node.getTextContent());
            if (node.getNextSibling() != null)
            {
                gStack.push(node.getNextSibling());
                domTraversor(node.getNextSibling());
            }
            else
            {
                if (node.getParentNode().getNextSibling() != null)
                    domTraversor(node.getParentNode().getNextSibling());
            }
        }
        else
        {
            if (node.getChildNodes().getLength() > 1)
            {
                gStack.push(node);
                Node n = node.getFirstChild();
                if (n.getNodeName().equals("#text"))
                {
                    System.out.println("textElem = " + n.getTextContent());
                    if (n.getNextSibling() != null)
                    {
                        gStack.push(n.getNextSibling());
                        domTraversor(n.getNextSibling());
                    }
                }
                else
                {
                    gStack.push(n);
                    domTraversor(n);
                }
            }
            else if (node.getChildNodes().getLength() == 1)
            {
                Node fnode = node.getFirstChild();
                if (fnode.getChildNodes().getLength() > 1)
                {
                    gStack.push(fnode);
                    domTraversor(fnode);
                }
                else
                {
                    if (!fnode.getNodeName().equals("#text"))
                    {
                        gStack.push(fnode);
                        domTraversor(fnode);
                    }
                    else
                    {
                        System.out.println("textElem = " + fnode.getTextContent());
                        if (fnode.getNodeName().equals("#text"))
                        {
                            if (node.getNextSibling() != null)
                            {
                                gStack.push(node.getNextSibling());
                                domTraversor(node.getNextSibling());
                            }
                            else
                            {
                                if (!gStack.empty())
                                {
                                    Node sibPn = gStack.pop();
                                    if (sibPn.getNextSibling() == null)
                                    {
                                        sibPn = gStack.pop();
                                    }
                                    domTraversor(sibPn.getNextSibling());
                                }
                            }
                        }
                        else
                        {
                            if (fnode.getNextSibling() != null)
                            {
                                domTraversor(fnode.getNextSibling());
                            }
                            else
                            {
                                if (!gStack.empty())
                                {
                                    Node sibPn = gStack.pop().getNextSibling();
                                    domTraversor(sibPn);
                                }
                            }
                        }
                    }
                }
            }
        }
    }
}
一部のxmlドキュメントでは正常に機能しますが、タグのようなドキュメントでは機能しません。
<unorderedlist>
    <item>
        <paragraph>
            <emphasis>alternaterowcolor</emphasis>
            , do all rows have the same color, or should the background
            color
            alternate?
        </paragraph>
    </item>
    <item>
        <paragraph>
            <emphasis>border</emphasis>
            , a limited choice of border styles.
        </paragraph>
    </item>
    <item>
        <paragraph>
            <emphasis>color</emphasis>
            , a limited choice of colors.
        </paragraph>
    </item>
</unorderedlist>
これは、要素に 3 つ以上のネストされた子がある場合、コードが停止し、先に進まないシナリオです。
より良い実装がある人はいますか、提案してください。