JAVA,NodeList XML让所有孩子都不知道te XML的内容

时间:2017-03-30 10:44:03

标签: java xml tree vaadin children

我正在使用Java框架Vaadin。

我使用XML内容,我想用这些信息创建一个树,但我不知道XML内容将持续多长时间。我发送一个调用来获取XML,但使用不同的IP地址。

我知道如何让孩子和下一个孩子再来。但是我希望自动化这个函数,比如,检查XML的持续时间并循环直到结束。

现在我的代码:

private void getData(NodeSet data)
{
    InputStream dataXml = new ByteArrayInputStream(data.toXMLString().getBytes(StandardCharsets.UTF_8));
    final DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
    try
    {
        final DocumentBuilder builder = factory.newDocumentBuilder();
        final Document document = builder.parse(dataXml);
        final Element racine = document.getDocumentElement();
        final NodeList racineNoeuds = racine.getChildNodes();
        final int nbRacineNoeuds = racineNoeuds.getLength();

        for (int i = 0; i < nbRacineNoeuds; i++)
        {
                    //System.out.print(nbRacineNoeuds);
            //System.out.print(racineNoeuds);
            if (racineNoeuds.item(i).getNodeType() == Node.ELEMENT_NODE)
            {
                final Element child = (Element) racineNoeuds.item(i);
                final NodeList racineChild = child.getChildNodes();
                final int nbRacineChild = racineChild.getLength();

                for (int y = 0; y < nbRacineChild; y++)
                {
                    if (racineChild.item(y).getNodeType() == Node.ELEMENT_NODE)
                    {
                        final Element child2 = (Element) racineChild.item(y);
                        final NodeList children = child2.getChildNodes();
                        final int nbChildren = children.getLength();

                        for (int a = 0; a < nbChildren; a++)
                        {
                            if (children.item(a).getNodeType() == Node.ELEMENT_NODE)
                            {

                                final Element child3 = (Element) children.item(y);

                            }
                        }
                    }
                }
                        //System.out.print(child);

                //tree.setParent(racineChild, child);
            }
            /*tree.addItem(racine);
             tree.addItem(racineNoeuds);
             tree.setParent(racineNoeuds, racine);
             tree.addItem(child);
             tree.setParent(child, racine);
             tree.addItem(child2);
             tree.setParent(child2, child);
             tree.addItem(child3);
             tree.setParent(child3, child2);*/
        }
    } catch (final ParserConfigurationException e)
    {
        e.printStackTrace();
    } catch (final SAXException e)
    {
        e.printStackTrace();
    } catch (final IOException e)
    {
        e.printStackTrace();
    }

}

谢谢!

1 个答案:

答案 0 :(得分:1)

由于我的答案来得相当晚,我猜这主要是为了后人:)

事实上,正如@Morfic指出的那样,递归方法可能是您现在可以实现的最简单的方法。

我创建了一个tiny project来说明一个非常基本的解决方案。它包含一个DataSource类,提供一些测试数据和一个main Vaadin UI,其内容为Tree,由您需要的 递归方法填充

/**
 * If <code>parentId</code> is <code>null</code> then the <code>element</code>'s
 * name or value gets added directly under <code>tree</code>.
 */
private void loadTree(Tree tree, Object parentId, Node element) {
    if (tree == null || element == null) {
        return;
    }

    // 1. We add to the tree what's given to us as argument:

    final String itemId =
        (element.getNodeType() == Node.TEXT_NODE ? element.getNodeValue() : element.getNodeName()).trim();
    if (itemId.length() == 0) {
        return; // skip eventual white spaces...
    }
    tree.addItem(itemId); // a tree item's label defaults to its ID

    // 2. If a parent ID is also provided as argument, we set that as parent for the newly added item

    if (parentId != null) {
        tree.setParent(itemId, parentId);
    }

    // 3. We then _recursively_ add to the tree every child of the given element (if any):

    NodeList children = element.getChildNodes();
    for (int i = 0, n = children.getLength(); i < n; i++) {
        loadTree(tree, itemId, children.item(i));
    }

    // 4. Optionally we can expand an item if it has any children or force hiding the arrow if not:

    if (children.getLength() > 0) {
        tree.expandItem(itemId);
    } else {
        tree.setChildrenAllowed(itemId, false);
    }
}

该方法首先向提供的Tree添加一个项目,其标签/ ID是根据Node的内容或名称创建的。如果还给出了父ID,我们将其设置为新添加节点的父级。然后,如果给定的Node有任何子节点,我们递归调用每个子节点的方法,将新创建的项目的ID作为父节点。

这应该可以解决问题。话虽如此,你应该(重新)考虑一些事情:

  1. 虽然简单,但如果XML(非常)很大,递归实现可能会让您遇到麻烦;然后你可以选择相同方法的迭代实现(一旦提供了递归方法,通常不是很难获得)

  2. 显然,第一次调用时,parentId参数可以/应该保留null。由于这往往会使代码更难理解(通常,null参数不是很清楚),您可以使用private void loadTree(Tree tree, Node element) { loadTree(tree, null, element); }重载方法,在其中隐藏丑陋的参数;只是对呼叫者友好... ...

  3. 我显然在UI中直接使用了XML DOM模型/ API - 通常不是最好的想法(即将应用程序的UI与低级模型联系起来);在一个更复杂的应用程序中,我们确实有更好的模型(领域模型,UI模型等);同样,人们可能想使用HierarchicalContainer

  4. Tree组件(还)在Vaadin 8中没有等价物;因此,该项目仅适用于Vaadin版本(不包括)8

  5. Bon勇气alors;)