为什么使用Java在文件的开头和结尾追加字符串后,xml文件无法正确对齐?

萨沙亚

我已在XML File的开头和结尾添加了字符串。但是在得到结果后,对齐方式不正确。

我的XML文件:

        import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.Reader;

import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import javax.xml.parsers.ParserConfigurationException;
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 org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.w3c.dom.NamedNodeMap;
import org.w3c.dom.Node;
import org.w3c.dom.NodeList;
import org.w3c.dom.Text;
import org.xml.sax.SAXException;

import com.google.common.io.Resources;

import java.io.StringReader;
import java.io.StringWriter;
import java.net.URL;
import java.nio.charset.Charset;

import javax.xml.transform.OutputKeys;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.TransformerFactoryConfigurationError;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;

public class ModifyXMLFile {

    public static void main(String args[]) {

        try {
            // Point the file directory path here
            String directory = "C:\\Users\\n444479\\Desktop\\SA";
            int test = new File("C:\\Users\\n444479\\Desktop\\SA").listFiles().length;
            File[] files = new File(directory).listFiles();

            // Loop the file to run all the XML files
            for (int j = 0; j < test; j++) {
                System.out.println(files[j]);

                String filepathext = files[j].toString();

                DocumentBuilderFactory docFactory = DocumentBuilderFactory
                        .newInstance();
                DocumentBuilder docBuilder = docFactory.newDocumentBuilder();
                Document doc = docBuilder.parse(filepathext);

                /*
                // XML file update starts here

                // 1.Add the attribute element with value between the two XML
                // Child elements



                 NodeList nodes = doc.getElementsByTagName("dummySegmentOne");
                 Text a = doc.createTextNode("value"); Element p =
                 doc.createElement("newNode"); p.appendChild(a);
                 nodes.item(0).getParentNode().insertBefore(p, nodes.item(0));


                // 2.Add the attribute element without value between the two XML
                // Child elements


                 NodeList nodesa =
                 doc.getElementsByTagName("customerLevelRegDocs"); Element q =
                 doc.createElement("dummySegmentOne");
                 nodesa.item(0).getParentNode().insertBefore(q,
                 nodesa.item(0));


                // 3.Rename the element in parent and child both using the JAXP
                // Parser
                */
                // XSLT File:

                String xsltResource = "C:\\Users\\n444479\\Desktop\\AB\\test.xml";

                StringWriter xmlResultResource = new StringWriter();

                Transformer xmlTransformer = TransformerFactory.newInstance().newTransformer(
                                new StreamSource(new File(xsltResource)));
                xmlTransformer.transform(new StreamSource(new File(filepathext)),new StreamResult(xmlResultResource));

                // XML file update end here

                // write the content into XML file
                TransformerFactory transformerFactory = TransformerFactory
                        .newInstance();
                Transformer transformer = transformerFactory.newTransformer();
                DOMSource source = new DOMSource(doc);
                StreamResult result = new StreamResult(new String(filepathext));
                transformer.transform(source, result);
                transformer.setOutputProperty(OutputKeys.INDENT, "yes");
                transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "2");
                // Successful output once get it "Done"
                String output = xmlResultResource.getBuffer().toString();
                // Writing the transformed XML to a file
                FileWriter fileWriter = new FileWriter(filepathext);
                fileWriter.write(output);
                fileWriter.close();

                System.out.println("Done");

            }
        }
        // Exception handling
        catch (ParserConfigurationException pce) {
            pce.printStackTrace();
        } catch (TransformerException tfe) {
            tfe.printStackTrace();
        } catch (IOException ioe) {
            ioe.printStackTrace();
        } catch (SAXException sae) {
            sae.printStackTrace();
        }
    }

}

程序执行完字符串后,将正确附加该字符串,但是XML File的对齐方式发生了变化。

我得到的输出如下

<?xml version="1.0" encoding="UTF-8"?>{% from lxml import etree %}{% from StringIO import StringIO %}{% set tree = parse_xml(request_text) %}{% set namespaces = {'soapenv': 'http://schemas.xmlsoap.org/soap/envelope/',  'wbs': 'http://xml.ama.com/ws/2009/01/WBS_Session-2.0.xsd'}%}<soap:Envelope xmlns:xalan="http://xml.apache.org/xalan" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"><soap:Header><Session><SessionId>{{tree.xpath('//soapenv:Header/wbs:Session/wbs:SessionId', namespaces=namespaces)[0].text}}</SessionId><SequenceNumber>{{int(tree.xpath('//soapenv:Header/wbs:Session/wbs:SequenceNumber', namespaces=namespaces)[0].text)+1}}</SequenceNumber><SecurityToken>{{tree.xpath('//soapenv:Header/wbs:Session/wbs:SecurityToken', namespaces=namespaces)[0].text}}</SecurityToken></Session></soap:Header><soap:Body><reRoot><reNode> world</reNode></reRoot></soap:Body></soap:Envelope>

为什么文件没有被附加

小桑蒂

原因是DataInputStream.readLine放弃换行符。因此,您必须自己添加它。

此外:考虑到生成的XML是不正确的,因为原来的XML标头竟然包含在一个节点中。标头(如果存在)必须始终位于根节点之前。

为了将输入的XML正确地转换为SOAP,我建议您两种选择:

  • 可以通过XSL样式表对其进行转换
  • 或者使用DOM构建结果XML,并将源XML作为一个子节点包含在内。然后,在序列化整个文档时,可以指定缩进参数。

更新

我建议您使用此XSL:

<?xml version="1.0" encoding="ISO-8859-1"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

<xsl:output method="xml" version="1.0" encoding="UTF-8"/>

<xsl:strip-space elements="*"/>

<xsl:template match="/">{% from lxml import etree %}{% from StringIO import StringIO %}{% set tree = parse_xml(request_text) %}{% set namespaces = {'soapenv': 'http://schemas.xmlsoap.org/soap/envelope/',  'wbs': 'http://xml.ama.com/ws/2009/01/WBS_Session-2.0.xsd'}%}<soap:Envelope xmlns:xalan="http://xml.apache.org/xalan" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/">
  <soap:Header>
    <Session>
      <SessionId>{{tree.xpath('//soapenv:Header/wbs:Session/wbs:SessionId', namespaces=namespaces)[0].text}}</SessionId>
      <SequenceNumber>{{int(tree.xpath('//soapenv:Header/wbs:Session/wbs:SequenceNumber', namespaces=namespaces)[0].text)+1}}</SequenceNumber>
      <SecurityToken>{{tree.xpath('//soapenv:Header/wbs:Session/wbs:SecurityToken', namespaces=namespaces)[0].text}}</SecurityToken>
    </Session>
  </soap:Header>
  <soap:Body><xsl:copy-of select="."/>
</soap:Body>
</soap:Envelope>
</xsl:template>

</xsl:stylesheet>

换句话说:它创建具有固定格式的SOAP消息,并将输入的xml设置为<soap:Body>节点的内容

以及转换代码:

private static void transform(org.w3c.dom.Document doc, java.io.InputStream inputXsl, java.io.OutputStream out)
    throws java.io.IOException,
    javax.xml.transform.TransformerConfigurationException,
    javax.xml.transform.TransformerException
{
    javax.xml.transform.Templates templates=javax.xml.transform.TransformerFactory.newInstance().newTemplates(new javax.xml.transform.stream.StreamSource(inputXsl));
    javax.xml.transform.Transformer transformer=templates.newTransformer();
    transformer.setOutputProperty(OutputKeys.INDENT, "yes");
    transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "4");
    javax.xml.transform.Result result=new javax.xml.transform.stream.StreamResult(out);
    javax.xml.transform.Source source=new javax.xml.transform.dom.DOMSource(doc);
    if (doc.getInputEncoding() != null)
    {
        transformer.setOutputProperty("encoding", doc.getInputEncoding());
    }
    transformer.transform(source, result);
}

整个结果通过两个原因重新缩进:

  • xsl:strip-space在XSL。
  • 转换中的indentindent-amount属性。

本文收集自互联网,转载请注明来源。

如有侵权,请联系 [email protected] 删除。

编辑于
0

我来说两句

0 条评论
登录 后参与评论

相关文章

在读取文件的字符串的开头和结尾处追加一些字符串

在 C 中对齐字符串的开头和结尾

如何在文件中以两个特定字符开头和结尾的字符串grep?

Perl解析JavaScript文件regex,仅在返回的字符串的开头和结尾捕获引号

下载JSON文件中以不同字符串开头(包括和结尾)的URL

删除文件中所有以特定字符串开头和结尾的行

通过开头和结尾的字符串来缩短可变长文件名

使用 bash 在行尾追加字符串

使用右对齐的字符串创建Java文件

为什么在使用 ofstream 变量向文件添加一些信息后,无法将文件中的数据写入字符串?

查找并替换以新字符串java开头和结尾的子字符串

如何删除某些字符后的字符串开头和结尾?

在 XML 文件中查找和修剪字符串结尾

将字符添加到文件中各种长度字符串的开头/结尾(Python / * nix)

为什么使用Java扫描仪从txt文件读取的字符串不等于相同的字符串?

为什么在Java中字符串以“”开头?

如何在字符串的开头和结尾添加字符?

从字符串的开头和结尾删除非字母字符

Javascript从字符串的开头和结尾删除特殊字符

匹配字符串,除非以字符开头和结尾

如何替换字符串的开头和结尾的特定字符?

MySQL函数从字符串的开头和结尾删除字符

在特定字符串的开头和结尾替换字符

查找以特定字符开头和结尾的字符串

从字符串的开头和结尾删除特殊字符

在文件中每行的开头和中间添加字符串

Java:从字符串的开头和结尾删除固定数量的字符

什么正则表达式会捕获字符串的开头和结尾?

使用正则表达式查找以字符开头和结尾的字符串组