我正在使用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();
}
}
谢谢!
答案 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作为父节点。
这应该可以解决问题。话虽如此,你应该(重新)考虑一些事情:
虽然简单,但如果XML(非常)很大,递归实现可能会让您遇到麻烦;然后你可以选择相同方法的迭代实现(一旦提供了递归方法,通常不是很难获得)
显然,第一次调用时,parentId
参数可以/应该保留null
。由于这往往会使代码更难理解(通常,null
参数不是很清楚),您可以使用private void loadTree(Tree tree, Node element) { loadTree(tree, null, element); }
重载方法,在其中隐藏丑陋的参数;只是对呼叫者友好... ...
我显然在UI中直接使用了XML DOM模型/ API - 通常不是最好的想法(即将应用程序的UI与低级模型联系起来);在一个更复杂的应用程序中,我们确实有更好的模型(领域模型,UI模型等);同样,人们可能想使用HierarchicalContainer
Tree
组件(还)在Vaadin 8中没有等价物;因此,该项目仅适用于Vaadin版本(不包括)8
Bon勇气alors;)