JSON4S类型提示不起作用

时间:2017-10-19 15:55:44

标签: json scala serialization json4s hive-serde

以下测试代码段

implicit val formats = DefaultFormats + FullTypeHints(Contacts.classList)

val serialized = Serialization.write(List(Mail(field = "random@mail.com", note = "Random note.")))
println(serialized)

Serialization.read[List[Contact[_]]](serialized).isInstanceOf[List[Mail]] should be (true)

失败
Can't find constructor for Contact[Object]
org.json4s.package$MappingException: Can't find constructor for Contact[Object]
    at org.json4s.reflect.package$.fail(package.scala:95)
    at org.json4s.reflect.ScalaSigReader$$anonfun$5.apply(ScalaSigReader.scala:21)
    at org.json4s.reflect.ScalaSigReader$$anonfun$5.apply(ScalaSigReader.scala:21)
    at scala.Option.getOrElse(Option.scala:121)
    at org.json4s.reflect.ScalaSigReader$.readConstructor(ScalaSigReader.scala:21)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder.ctorParamType(Reflector.scala:93)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder$$anonfun$createConstructorDescriptors$3$$anonfun$15.apply(Reflector.scala:156)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder$$anonfun$createConstructorDescriptors$3$$anonfun$15.apply(Reflector.scala:142)
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:234)
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:234)
    at scala.collection.mutable.ResizableArray$class.foreach(ResizableArray.scala:59)
    at scala.collection.mutable.ArrayBuffer.foreach(ArrayBuffer.scala:48)
    at scala.collection.TraversableLike$class.map(TraversableLike.scala:234)
    at scala.collection.AbstractTraversable.map(Traversable.scala:104)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder$$anonfun$createConstructorDescriptors$3.apply(Reflector.scala:142)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder$$anonfun$createConstructorDescriptors$3.apply(Reflector.scala:136)
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:234)
    at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:234)
    at scala.collection.mutable.ArraySeq.foreach(ArraySeq.scala:74)
    at scala.collection.TraversableLike$class.map(TraversableLike.scala:234)
    at scala.collection.AbstractTraversable.map(Traversable.scala:104)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder.createConstructorDescriptors(Reflector.scala:136)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder.constructorsAndCompanion(Reflector.scala:121)
    at org.json4s.reflect.Reflector$ClassDescriptorBuilder.result(Reflector.scala:183)
    at org.json4s.reflect.Reflector$.createDescriptor(Reflector.scala:53)
    at org.json4s.reflect.Reflector$$anonfun$describe$1.apply(Reflector.scala:48)
    at org.json4s.reflect.Reflector$$anonfun$describe$1.apply(Reflector.scala:48)
    at org.json4s.reflect.package$Memo.apply(package.scala:36)
    at org.json4s.reflect.Reflector$.describe(Reflector.scala:48)
    at org.json4s.Extraction$$anonfun$extract$6.apply(Extraction.scala:393)
    at org.json4s.Extraction$$anonfun$extract$6.apply(Extraction.scala:392)
    at org.json4s.Extraction$.customOrElse(Extraction.scala:606)
    at org.json4s.Extraction$.extract(Extraction.scala:392)
    at org.json4s.Extraction$CollectionBuilder$$anonfun$7.apply(Extraction.scala:410)
    at org.json4s.Extraction$CollectionBuilder$$anonfun$7.apply(Extraction.scala:410)
    at scala.collection.immutable.List.map(List.scala:284)
    at org.json4s.Extraction$CollectionBuilder.mkCollection(Extraction.scala:410)
    at org.json4s.Extraction$CollectionBuilder.result(Extraction.scala:430)
    at org.json4s.Extraction$$anonfun$extract$5.apply(Extraction.scala:382)
    at org.json4s.Extraction$$anonfun$extract$5.apply(Extraction.scala:382)
    at org.json4s.Extraction$.customOrElse(Extraction.scala:606)
    at org.json4s.Extraction$.extract(Extraction.scala:382)
    at org.json4s.Extraction$.extract(Extraction.scala:39)
    at org.json4s.ExtractableJsonAstNode.extract(ExtractableJsonAstNode.scala:21)
    at org.json4s.jackson.Serialization$.read(Serialization.scala:50)
    at org.json4s.Serialization$class.read(Serialization.scala:30)
    at org.json4s.jackson.Serialization$.read(Serialization.scala:17)

其中Contact

abstract class Contact[Field : Validable](
  val field: Field,
  val created: Long,
  val updated: Long,
  val note: String) { }

Mail

case class Mail(
  override val field: String,
  override val created: Long = System.currentTimeMillis(),
  override val updated: Long = System.currentTimeMillis(),
  override val note: String)
extends Contact[String](field, created, updated, note)(Mail)

case object Mail extends Validable[String] {
  override def valid(field: String): Boolean = {
    Validator.email(field)
  }
}

并且测试输出是

[{"jsonClass":"whatever.core.entities.utility.contact.Mail","field":"random@mail.com","created":1508428385266,"updated":1508428385266,"note":"Random note."}]

深入研究JSON4S的代码库,表明根本没有使用类型提示。

如何强制JSON4S使用类型提示?

干杯

2 个答案:

答案 0 :(得分:0)

这是您如何使用FullTypeHints的示例。

您不能隐式询问Validable[Field],也不能添加多个显式参数列表(这也适用于隐式参数,请参阅下面的注释),因为JSON4S不支持它们。

注意:使用Validable[Field]语法隐式请求Contact[Field: Validable]等同于使用implicit validable: Validable[Field]添加其他参数列表。

您可以将另一个参数validable: Validable[Field]添加到Contact构造函数或(如下例所示)一个validable字段,该字段应由具体类重写(如{ {1}})扩展Mail抽象类。

Contact
trait Validable[T]{
  def valid(field: T): Boolean
}

abstract class Contact[Field](
  val field: Field, val created: Long, 
  val updated: Long, val note: String) {
  val validable: Validable[Field]
}

object Contacts{ val classList = List(classOf[Mail]) }

case class Mail(
  override val field: String,
  override val created: Long = System.currentTimeMillis(),
  override val updated: Long = System.currentTimeMillis(),
  override val note: String
) extends Contact[String](field, created, updated, note){
  override val validable: Validable[String] = Mail
}

case object Mail extends Validable[String] {
  override def valid(field: String): Boolean = true
}

implicit val formats = DefaultFormats + FullTypeHints(Contacts.classList)

val mail: List[Mail] = List(Mail(field = "random@mail.com", note = "Random note.")) val serialized = Serialization.write(mail) val mailS = Serialization.read[List[Contact[_]]](serialized) print(mail == mailS) JSON表示如下:

serialized

答案 1 :(得分:0)

使用Contact实施trait,它会起作用。