Как удалить весь узел в простом элементе с помощью xpath и Java

У меня есть такой xml:

<?xml version = "1.0" encoding = "UTF-8" standalone = "yes"?>
<User SOURCE_NAME = "PublicAssetFeed" xmlns = "http://abc.e.wat.com/xml">
<Employee>
<FIELD NAME = "Name" TYPE = "char">Rahul</FIELD>
<FIELD NAME = "Branch" TYPE = "char"></FIELD>
<FIELD NAME = "Unique ID" TYPE = "char">12345</FIELD>
</Employee> `

Я хочу удалить всю строку где НАЗВАНИЕ="Филиал". Итак, мой окончательный XML должен выглядеть так

`<?xml version = "1.0" encoding = "UTF-8" standalone = "yes"?>
<User SOURCE_NAME = "PublicAssetFeed" xmlns = "http://abc.e.wat.com/xml">
<Employee>
<FIELD NAME = "Name" TYPE = "char">Rahul</FIELD>
<FIELD NAME = "Unique ID" TYPE = "char">12345</FIELD>
</Employee>
</User>`

Мне нужно сделать это с помощью Java. Вышеупомянутый XML хранится как строка. Итак, мне нужно преобразовать его в XML, удалить конкретную строку, а затем преобразовать обратно в строку. Я попытался использовать XPath, и я смог найти только основной узел, который является «ПОЛЕ», используя приведенный ниже код Java. Как удалить всю строку, где NAME="Branch"

package com.javamultiplex;
import java.io.IOException;
import java.io.StringReader;
import org.w3c.dom.Node;
import org.xml.sax.SAXException;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.xpath.XPath;
import javax.xml.xpath.XPathConstants;
import javax.xml.xpath.XPathExpressionException;
import javax.xml.xpath.XPathFactory;

import org.xml.sax.InputSource;

public class test {

    public static void main(String[] args) throws SAXException, IOException, ParserConfigurationException, XPathExpressionException{    
        String abc = "<?xml version=\"1.0\" encoding=\"UTF-8\" standalone=\"yes\"?><User SOURCE_NAME=\"PublicAssetFeed\" xmlns=\"http://abc.e.wat.com/xml\"><Employee><FIELD NAME=\"Name\" TYPE=\"char\">Rahul</FIELD><FIELD NAME=\"Unique ID\" TYPE=\"char\">12345</FIELD><FIELD NAME=\"Branch\" TYPE=\"char\"></FIELD></Employee></User>"
        DocumentBuilderFactory docFactory = DocumentBuilderFactory.newInstance();
        DocumentBuilder docBuilder = docFactory.newDocumentBuilder();
        XPath xPath = XPathFactory.newInstance().newXPath();
        InputSource sourceMasterTitle = new InputSource(new StringReader(abc.toString()));
        String expression = "//FIELD[@NAME='Branch']";;         
        Node value = (Node) xPath.evaluate(expression, sourceMasterTitle ,XPathConstants.NODE);   
        System.out.println("Main node is "+value.getNodeName());
    }
}

Удаление узла из XML означает, что мы удаляем элемент из его родителя. Для этого вы можете использовать метод removeChild() родительского узла узла.

Kris 01.07.2019 10:26

Похоже, вы были на правильном пути со своим XPath, я не понимаю, с какой проблемой вы столкнулись. Вы напечатали value.getNodeName(), поэтому у вас было только FIELD, но удаление этого узла, вероятно, привело бы к ожидаемому поведению.

Aaron 01.07.2019 11:35
Пользовательский скаляр GraphQL
Пользовательский скаляр GraphQL
Листовые узлы системы типов GraphQL называются скалярами. Достигнув скалярного типа, невозможно спуститься дальше по иерархии типов. Скалярный тип...
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
Как вычислять биты и понимать побитовые операторы в Java - объяснение с примерами
В компьютерном программировании биты играют важнейшую роль в представлении и манипулировании данными на двоичном уровне. Побитовые операции...
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Поднятие тревоги для долго выполняющихся методов в Spring Boot
Приходилось ли вам сталкиваться с требованиями, в которых вас могли попросить поднять тревогу или выдать ошибку, когда метод Java занимает больше...
Полный курс Java для разработчиков веб-сайтов и приложений
Полный курс Java для разработчиков веб-сайтов и приложений
Получите сертификат Java Web и Application Developer, используя наш курс.
0
2
465
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Файлы XML представляют собой гибкие и интеллектуальные структуры, основанные на древовидной структуре. Было бы лучше спроектировать свой код так, чтобы он проходил через узлы и удалял тот, который вы хотите удалить.

Можно изменить отступ, изменить структуру XML-файла. Если у вас есть подход типа «удаление строки», ваш код может устареть.

Вместо этого вам нужно преобразовать строку в документ и пройтись по его узлам.

package stackoverflow;

import java.io.IOException;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.io.StringReader;

import org.w3c.dom.Document;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.xml.sax.SAXException;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
import javax.xml.transform.OutputKeys;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.dom.DOMSource;
import javax.xml.transform.stream.StreamResult;
import javax.xml.xpath.XPathExpressionException;

import org.xml.sax.InputSource;

public class XMLTester {

    public static void main(String[] args)
            throws SAXException, IOException, ParserConfigurationException, XPathExpressionException {
        String xmlStr = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>"
                + "<User xmlns=\"http://abc.e.wat.com/xml\" SOURCE_NAME=\"PublicAssetFeed\">" + "   <Employee>"
                + "      <FIELD NAME=\"Name\" TYPE=\"char\">Rahul</FIELD>"
                + "      <FIELD NAME=\"Branch\" TYPE=\"char\" />"
                + "      <FIELD NAME=\"Unique ID\" TYPE=\"char\">12345</FIELD>" + "   </Employee>" + "</User>";

        Document doc = readStringAsDocument(xmlStr);

        try {
            printDocument(doc, System.out);
        } catch (TransformerException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }

        Node willBeDeleted = null;

        NodeList fieldNodes = doc.getElementsByTagName("FIELD");

        if (fieldNodes.getLength() > 0) {
            for (int i = 0; i < fieldNodes.getLength(); i++) {
                Node node = fieldNodes.item(i);
                NamedNodeMap attributes = node.getAttributes();

                for (int j = 0; j < attributes.getLength(); j++) {
                    Node att = attributes.item(j);

                    if (att.getNodeName().equals("NAME") && att.getNodeValue().equals("Branch")) {
                        willBeDeleted = node;
                    }
                }
            }
        }

        willBeDeleted.getParentNode().removeChild(willBeDeleted);

        try {
            printDocument(doc, System.out);
        } catch (TransformerException e) {
            // TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    private static Document readStringAsDocument(String xmlString)
            throws ParserConfigurationException, SAXException, IOException {
        DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();

        DocumentBuilder db = null;

        db = dbf.newDocumentBuilder();

        Document doc = db.parse(new InputSource(new StringReader(xmlString)));
        return doc;
    }

    // https://stackoverflow.com/a/2325407/2384806
    public static void printDocument(Document doc, OutputStream out) throws IOException, TransformerException {
        TransformerFactory tf = TransformerFactory.newInstance();
        Transformer transformer = tf.newTransformer();
        transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "no");
        transformer.setOutputProperty(OutputKeys.METHOD, "xml");
        transformer.setOutputProperty(OutputKeys.INDENT, "yes");
        transformer.setOutputProperty(OutputKeys.ENCODING, "UTF-8");
        transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "4");

        transformer.transform(new DOMSource(doc), new StreamResult(new OutputStreamWriter(out, "UTF-8")));
    }
}

Большое спасибо @mahmutoflaz. Именно то, что я искал.

Pavan Kumar 01.07.2019 15:11

Я принял ответ ... но для голосования мне нужно иметь 15 репутаций, которых у меня нет. Еще раз спасибо @mahmutoflaz

Pavan Kumar 02.07.2019 13:00

Другие вопросы по теме