1.載入Xml
從檔案載入
SAXReader reader = new SAXReader(); String filePath = "/xmlfile/" + fileName + ".xml"; Document document = null; try { document = reader.read(ProcessXmlUtil.class.getResourceAsStream(filePath)); } catch (DocumentException e) { logger.error("error:", e); } return document;
通過字串轉換
Document document = DocumentHelper.parseText(xmlString);
2.驗證
/** * 通過XSD驗證xml的準確性 * * @param xsdFileName xsd路徑 要全路徑 * @param xmlString xml檔案 * @return */ public static String validateXsd(String xsdFileName, String xmlString) { try { XMLErrorHandler handler = new XMLErrorHandler(); //獲取基於 SAX 的解析器的例項 SAXParserFactory factory = SAXParserFactory.newInstance(); //解析器在解析時驗證 XML 內容。 factory.setValidating(true); //指定由此程式碼生成的解析器將提供對 XML 名稱空間的支援。 factory.setNamespaceAware(true); //使用當前配置的工廠引數建立 SAXParser 的一個新例項。 SAXParser parser = factory.newSAXParser(); //建立一個讀取工具 //獲取要校驗xml文件例項 Document document = DocumentHelper.parseText(xmlString); parser.setProperty( "http://java.sun.com/xml/jaxp/properties/schemaLanguage", "http://www.w3.org/2001/XMLSchema"); parser.setProperty( "http://java.sun.com/xml/jaxp/properties/schemaSource", "file:" + xsdFileName); //建立一個SAXValidator校驗工具,並設定校驗工具的屬性 SAXValidator validator = new SAXValidator(parser.getXMLReader()); //設定校驗工具的錯誤處理器,當發生錯誤時,可以從處理器物件中得到錯誤資訊。 validator.setErrorHandler(handler); //校驗 validator.validate(document); XMLWriter writer = new XMLWriter(OutputFormat.createPrettyPrint()); if (handler.getErrors().hasContent()) { logger.error("XML檔案通過XSD檔案校驗失敗!"); writer.write(handler.getErrors()); String error = ""; for (Node node : handler.getErrors().content()) { error += node.asXML() + System.getProperty("line.separator"); } return error; } else { logger.info("XML檔案通過XSD檔案校驗成功!"); } return ""; } catch (Exception ex) { logger.error("XML檔案通過XSD檔案:" + xsdFileName + "檢驗失敗。\n原因: " + ex.getMessage()); return ex.getMessage(); } }
注意,驗證xsd的路徑名要是完整的全路徑,如e://xml/xsd/test.xsd
3.Xpath
<?xml version="1.0" encoding="utf-8" ?> <root> <name value="張三"> <address>福建省廈門市XXX</address> </name> </root>
獲取值:福建省廈門市XXX
document.selectSingleNode("//name/address").getText();
獲取name的屬性value
document.selectSingleNode("//name/@value").getText();
名稱空間的處理:當XML有名稱空間時,Xpath的查詢也要加上名稱空間,且只能查詢一個層級
Map map = new HashMap(); map.put("ns", nameSpace); XPath x = newXmlDoc.createXPath("//ns:name");
這裡只查詢一級name是有值的,但是如果想同時查詢name下面的address,則返回的是空
Map map = new HashMap(); map.put("ns", nameSpace); XPath x = newXmlDoc.createXPath("//ns:name"); x.setNamespaceURIs(map); Node node= (x.selectSingleNode(rootNode));//有值 x = newXmlDoc.createXPath("//ns:name/address"); Node node2= (x.selectSingleNode(rootNode));//值為空
如果想查詢address上的值,則必須在name節點上查詢,如
Map map = new HashMap(); map.put("ns", nameSpace); XPath x = newXmlDoc.createXPath("//ns:name"); x.setNamespaceURIs(map); Node node= (x.selectSingleNode(rootNode));//有值 x = newXmlDoc.createXPath("//ns:address"); x.setNamespaceURIs(map); Node node2= (x.selectSingleNode(node));//有值
所以想一次性用Xpath查詢,最好先把名稱空間去掉
4.增加名稱空間
Document newXmlDoc = DocumentHelper.createDocument(); //頭部構建 Element rootNode = newXmlDoc.addElement(strMsgModelCode); rootNode.addNamespace("", nameSpace);
第一個引數是空
5.在固定節點插入
要迴圈遍歷節點,然後呼叫elements.add方法。這個真沒有.net的linq to xml方便。
List<Element> elements = rootNode.elements(); int index = 0; for (Element element : elements) { index++; if ("name".equals(element.getName())) { elements.add(index, responseEle); break; } }