Come emettere una sezione CDATA da un Sax XmlHandler
Domanda
Questa è una domanda di seguito di Come codificare i caratteri da Oracle a Xml?
Nel mio ambiente qui uso Java per serializzare il set di risultati su xml. Non ho accesso al flusso di output stesso, solo a un org.xml.sax.ContentHandler.
Quando provo ad emettere caratteri in una sezione CDATA:
Succede sostanzialmente così:
xmlHandler.startElement(uri, lname, "column", attributes);
String chars = "<![CDATA["+rs.getString(i)+"]]>";
xmlHandler.characters(chars.toCharArray(), 0, chars.length());
xmlHandler.endElement(uri, lname, "column");
Ottengo questo:
<column><![CDATA[33665]]></column>
Ma voglio questo:
<column><![CDATA[33665]]></column>
Quindi, come posso generare una sezione CDATA con Sax ContentHandler?
Soluzione
Si sta salvando perché la funzione handler.characters è progettata per scappare e la <![CDATA[
parte non è considerata parte del valore.
È necessario utilizzare i metodi appena esposti in DefaultHandler2
o utilizzare l'approccio TransformerHandler
in cui è possibile impostare la chiave di output CDATA_SECTION_ELEMENTS
, che accetta un elenco delimitato da spazi bianchi di nomi di tag che dovrebbero generare sezioni di sottotesto racchiuse in CDATA.
StreamResult streamResult = new StreamResult(out);
SAXTransformerFactory tf = (SAXTransformerFactory) SAXTransformerFactory.newInstance();
TransformerHandler hd = tf.newTransformerHandler();
Transformer serializer = hd.getTransformer();
serializer.setOutputProperty(OutputKeys.CDATA_SECTION_ELEMENTS, "column");
hd.setResult(streamResult);
hd.startDocument();
hd.startElement("","","column",atts);
hd.characters(asdf,0, asdf.length());
hd.endElement("","","column");
hd.endDocument();
Altri suggerimenti
È necessario utilizzare startCDATA()
e endCData()
come delimitatori, ovvero
xmlHandler.startElement(uri, lname, "column", attributes);
xmlHandler.startCDATA();
String chars = rs.getString(i);
xmlHandler.characters(chars.toCharArray(), 0, chars.length());
xmlHandler.endCDATA();
xmlHandler.endElement(uri, lname, "column");