开发者

Save Properties to XML without header

Properties class has very nice methods storeToXml and loadFromXml. But store adds header, so xml lo开发者_StackOverflow中文版oks like this

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE properties SYSTEM "http://java.sun.com/dtd/properties.dtd">
 <properties>
  <entry key="key">value</entry>
 </properties>

I don't want to have this header, because i store it in database to make it xpath serchable. And also i want to load in to Properties object. Have anyone good ideas? Thanks


Neither XPath nor Properties.loadFromXml() care for the header. So this should work as it is.

If it's really a problem, then write the result to a StringWriter or ByteArrayOutputStream and remove anything before <properties>. But that might actually cause errors loading the XML since the doctype is now missing.


Try this way:

Properties p = new Properties();
p.put("A.a", "BB");
p.put("A.b", "BB");
ByteArrayOutputStream bout = new ByteArrayOutputStream();
p.storeToXML(bout, "Commnet!");
String sp = new String(bout.toByteArray());
sp = sp.substring(sp.indexOf('\n', sp.indexOf('\n') + 1) + 1);
System.out.println(sp);

It will give you following result:

<properties>
<comment>COmmnet!</comment>
<entry key="A.b">BB</entry>
<entry key="A.a">BB</entry>
</properties>


Just remove the header :

BufferedReader br = new BufferedReader(new FileReader(new File("filename")));
BufferedWriter wr = new BufferedWriter(new FileWriter(new File("filename_out")));
String line;
int counter = 0;

while((line=br.readLine())!= null){
    if(counter > 0){
    wr.write(line.trim());
    }
    counter++;
}
wr.close();
br.close();


You could transform this xml using a xslt that removes the doctype declaration and so store the output in database.

public class XMLTransform {

  public static void main(String args[]) {

    try {
      StreamSource source = new StreamSource("your xml");
      StreamSource stylesource = new StreamSource("your xslt");

      TransformerFactory factory = TransformerFactory.newInstance();
      Transformer transformer = factory.newTransformer(stylesource);

      transformer.setOutputProperty(OutputKeys.OMIT_XML_DECLARATION, "yes");

      StreamResult result = new StreamResult(new File("your xml output file"));
      transformer.transform(source, result);
    } catch (Exception e) {
      e.printStackTrace();
    }
  }
}


Given that properties.dtd is so trivial, write and parse the XML yourself.

Output:

  private static String toXml(Properties props) throws XMLStreamException {
    Writer buffer = new StringWriter();
    XMLStreamWriter xml = XMLOutputFactory.newFactory()
        .createXMLStreamWriter(buffer);
    xml.writeStartElement("properties");
    for (Map.Entry<Object, Object> entry : props.entrySet()) {
      xml.writeStartElement("entry");
      xml.writeAttribute("key", entry.getKey()
          .toString());
      xml.writeCharacters(entry.getValue()
          .toString());
      xml.writeEndElement();
    }
    xml.writeEndElement();
    return buffer.toString();
  }

Input:

  private static Properties fromXml(String xml) throws XPathException {
    Properties props = new Properties();
    NodeList entries =
        (NodeList) XPathFactory.newInstance()
            .newXPath()
            .evaluate("//entry", new InputSource(new StringReader(xml)),
                XPathConstants.NODESET);
    for (int i = 0; i < entries.getLength(); i++) {
      Node entry = entries.item(i);
      props.setProperty(entry.getAttributes()
          .getNamedItem("key")
          .getNodeValue(), entry.getTextContent());
    }
    return props;
  }

I'm assuming you want data as the default JDBC mapping - String.

0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜