以下测试代码段
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使用类型提示?
干杯
答案 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
,它会起作用。