Как мне заставить XStream красиво выводить списки Scala? Могу ли я написать собственный конвертер? - PullRequest
6 голосов
/ 18 ноября 2009

Этот код:

println(new XStream.toXML(List(1,2,3)))

производит этот XML:

<scala.coloncolon serialization="custom">
  <unserializable-parents/>
  <scala.coloncolon>
    <int>1</int>
    <int>2</int>
    <int>3</int>
    <scala.ListSerializeEnd/>
  </scala.coloncolon>
</scala.coloncolon>

Вместо этого я хотел бы это:

<list>
  <int>1</int>
  <int>2</int>
  <int>3</int>
</list>

Что будет похоже на сериализацию обобщенных коллекций Java. Какой лучший способ сделать это?

Большую часть пути я проделал, реализовав свой собственный конвертер, но я застрял на методе unmarshal, не совсем понятно, как создать пустой список ...

class ListConverter( _mapper : Mapper )  extends AbstractCollectionConverter(_mapper) {
  /** Helper method to use x.getClass
   * 
   * See: http://scalide.blogspot.com/2009/06/getanyclass-tip.html
   */
  def getAnyClass(x: Any) = x.asInstanceOf[AnyRef].getClass

  def canConvert( clazz: Class[_]) = {       
    classOf[::[_]] == clazz
  }

  def marshal( value: Any, writer: HierarchicalStreamWriter, context: MarshallingContext) = {
    val list = value.asInstanceOf[List[_]]
    for ( item <- list ) {      
      writeItem(item, context, writer)
    }
  }

  def unmarshal( reader: HierarchicalStreamReader, context: UnmarshallingContext ) = {
    println(context.getRequiredType())
    var list : List[_] = createCollection(context.getRequiredType()).asInstanceOf[List[_]]
    while (reader.hasMoreChildren()) {
      reader.moveDown();
      val item = readItem(reader, context, list);
      list = item :: list
      reader.moveUp();
    }
    list
  }
}

object ListConverter {
  def configureXStream( stream: XStream ) = {
    stream.alias("list", classOf[::[_]])
    stream.registerConverter( new ListConverter(stream.getMapper) )        
  }
}

Ответы [ 2 ]

2 голосов
/ 18 ноября 2009

Не прошло и секунды после публикации вопроса, мне пришел ответ, вот рабочая реализация unmarshal:

  def unmarshal( reader: HierarchicalStreamReader, context: UnmarshallingContext ) = {
    var list : List[_] = Nil 
    while (reader.hasMoreChildren()) {
      reader.moveDown();
      val item = readItem(reader, context, list);
      list = list ::: List(item) // be sure to build the list in the same order
      reader.moveUp();
    }
    list
  }
1 голос
/ 18 ноября 2009

Существует только один экземпляр пустого списка, который является объектом Nil.

...