2016-08-11 1 views
1

Я пытаюсь сериализовать класс с использованием JAXB, который имеет некоторые поля CDATA, и некоторые поля, которые содержат специальные символы, которые необходимо экранировать (включая < и>). Проблема в том, что я не могу заставить обработчик escape работать правильно для обоих этих случаев.Escape XML-символы без экранирования метки CDATA

Использования CDATA адаптера Выборочного, если установка следующего свойства в моем ИАСЕ,

jaxbMarshaller.setProperty(CharacterEscapeHandler.class.getName(), 
     (CharacterEscapeHandler) (ch, start, length, isAttVal, out) -> out.write(ch, start, length)); 

я получаю:

<key1><![CDATA[Test]]></key1> # What I want 
<key2>some_>_value</key2>  # Invalid XML 

Если удалить эту собственность и пусть JAXB обрабатывать свое собственное вытекание, Я получаю:

<key1>&lt;![CDATA[Test]]&gt; # Not what I want 
<key2>some_&lt;_value</key2> # What I want 

Но что мне нужно:

<key1><![CDATA[Test]]></key1> 
<key2>some_&lt;_value</key2> 

Есть ли какие-либо способы, которыми я могу управлять функцией эвакуатора таким образом?

ответ

0

Вы можете использовать интерфейс com.sun.xml.bind.marshaller.CharacterEscapeHandler, чтобы устранить проблему.

Пример Java Класс:

import javax.xml.bind.annotation.XmlElement; 
import javax.xml.bind.annotation.XmlRootElement; 

@XmlRootElement 
public class AA { 

    String B; 
    String C; 


    public String getB() { 
     return B; 
    } 

    @XmlElement 
    public void setB(String b) { 
     B = b; 
    } 

    @Override 
    public String toString() { 
     return "AA [B=" + B + ", C=" + C + ", D=" + D + "]"; 
    } 

    public String getC() { 
     return C; 
    } 

    @XmlElement 
    public void setC(String c) { 
     C = c; 
    } 

    public String getD() { 
     return D; 
    } 

    @XmlElement 
    public void setD(String d) { 
     D = d; 
    } 

    String D; 
} 

Пример XML:

<?xml version="1.0" encoding="UTF-8" standalone="yes"?> 
<aa> 
    <b>Title of Feed Item</b> 
    <c>some_>_value</c> 
    <d> 
    <![CDATA[Test]]> 
    </d> 
</aa> 

Сериализация вызовов:

AA ref = new AA(); 
ref.setB("Title of Feed Item"); 
ref.setC("some_>_value"); 
ref.setD("<![CDATA[Test]]>"); 
File file = new File("Test.xml"); 
JAXBContext jaxbContext = JAXBContext.newInstance(AA.class); 
Marshaller jaxbMarshaller = jaxbContext.createMarshaller(); 
jaxbMarshaller.setProperty(Marshaller.JAXB_FORMATTED_OUTPUT, true); 
final Map<String, String> replacement=new HashMap<String, String>(); 
    replacement.put(">", "&gt;"); 
    replacement.put("<", "&lt;"); 
    final Pattern pattern=Pattern.compile("[<|>]"); 

jaxbMarshaller.setProperty(CharacterEscapeHandler.class.getName(), 
      new CharacterEscapeHandler() { 

       @Override 
       public void escape(char[] arg0, int arg1, int arg2, 
         boolean arg3, Writer arg4) throws IOException { 
        if (String.valueOf(arg0).contains("CDATA")) 
         arg4.write(arg0, arg1, arg2); 
        else { 
         StringBuffer buffer = new StringBuffer(); 
         Matcher matcher = pattern.matcher(String 
           .valueOf(arg0)); 
         while (matcher.find()) { 
          matcher.appendReplacement(buffer, 
            replacement.get(matcher.group())); 
         } 
         matcher.appendTail(buffer); 
         char t[] = buffer.toString().toCharArray(); 
         arg4.write(t, arg1, t.length); 
        } 
       } 
      }); 



jaxbMarshaller.marshal(ref, file); 
jaxbMarshaller.marshal(ref, System.out); 

Edit: Я изменил код с помощью регулярного выражения для скобок ,

Для регулярного образца выражения я упоминал this link

+0

Насколько я понимаю, это все-таки дать мне неправильный результат, так как ' некоторые _> _ значение' является недействительным XML. –