универсальная итеративная итерация NamedNodeMap - PullRequest
20 голосов
/ 13 ноября 2010

В Java, глядя на интерфейс NamedNodeMap , как вы выполняете его с обобщениями? Кажется, он использует Node, а не String, но я не совсем уверен, как использовать объекты Node ...

NamedNodeMap namedNodeMap = doc.getAttributes();
Map<String, String> stringMap = (Map<String, String>) namedNodeMap;
for (Map.Entry<String, String> entry : stringMap.entrySet()) {
  //key,value stuff here
}

Да, я могу видеть, как выполнять итерации без использования обобщений и с регулярным циклом for, но я бы хотел использовать приведенную выше «идиома» для карт. Конечно, проблема может заключаться в том, что, несмотря на название, NamedNodeMap фактически не реализует интерфейс Map! (

Полагаю, тебе просто нужно прикусить пулю и сделать что-то вроде:

/*
 * Iterates through the node attribute map, else we need to specify specific 
 * attribute values to pull and they could be of an unknown type
 */
private void iterate(NamedNodeMap attributesList) {
    for (int j = 0; j < attributesList.getLength(); j++) {
        System.out.println("Attribute: "
                + attributesList.item(j).getNodeName() + " = "
                + attributesList.item(j).getNodeValue());
    }
}

нет ничего лучше?

Ответы [ 4 ]

8 голосов
/ 20 февраля 2015

Вы можете создать свою собственную Iterable оболочку для NamedNodeMap и затем использовать ее в цикле foreach .

Например, это может быть простая реализация:

public final class NamedNodeMapIterable implements Iterable<Node> {

    private final NamedNodeMap namedNodeMap;

    private NamedNodeMapIterable(NamedNodeMap namedNodeMap) {
        this.namedNodeMap = namedNodeMap;
    }

    public static NamedNodeMapIterable of(NamedNodeMap namedNodeMap) {
        return new NamedNodeMapIterable(namedNodeMap);
    }

    private class NamedNodeMapIterator implements Iterator<Node> {

        private int nextIndex = 0;

        @Override
        public boolean hasNext() {
            return (namedNodeMap.getLength() > nextIndex);
        }
        @Override
        public Node next() {
            Node item = namedNodeMap.item(nextIndex);
            nextIndex = nextIndex + 1;
            return item;
        }

        @Override
        public void remove() {
            throw new UnsupportedOperationException();
        }

    }

    @Override
    public Iterator<Node> iterator() {
        return new NamedNodeMapIterator();
    }
}

В этом случае это будет использование:

private void iterate(NamedNodeMap attributesList) {
    for (Node node : NamedNodeMapIterable.of(attributesList)) {
        System.out.println("Attribute: "
                + node.getNodeName() + " = " + node.getNodeValue());
    }
}

С подобным подходом вы можете создать Iterable над Map.Entry<String, String> экземплярами.

7 голосов
/ 13 ноября 2010

Я не думаю, что есть более хороший способ использовать эти API. (Обновление: ОК - возможно https://stackoverflow.com/a/28626556/139985 считается хорошим).

Имейте в виду, что API-интерфейсы Java W3C DOM были заданы до того, как в Java появились общие или новый синтаксис for или даже интерфейс Iterator. Также имейте в виду, что API-интерфейсы W3C DOM для Java на самом деле являются результатом сопоставления спецификации IDL с Java.

Если вам нужны более приятные API для манипулирования XML и т. Д. В памяти, возможно, вам стоит взглянуть на JDOM.

4 голосов
/ 14 марта 2016

Так как вы не можете привести NamedNodeMap к Map , я предлагаю циклически использовать классический цикл for:

int numAttrs = namedNodeMap.getLength();
System.out.println("Attributes:");
for (int i = 0; i < numAttrs; i++){
   Attr attr = (Attr) pParameterNode.getAttributes().item(i);
   String attrName = attr.getNodeName();
   String attrValue = attr.getNodeValue();
   System.out.println("\t[" + attrName + "]=" + attrValue);
}
0 голосов
/ 11 апреля 2019

Из решения Java 8:

private static Iterable<Node> iterableNamedNodeMap(final NamedNodeMap namedNodeMap) {
    return () -> new Iterator<Node>() {

        private int index = 0;

        @Override
        public boolean hasNext() {
            return index < namedNodeMap.getLength();
        }

        @Override
        public Node next() {
            if (!hasNext())
                throw new NoSuchElementException();
            return namedNodeMap.item(index++);
        }
    };
}
...