Вы можете использовать бесформенное.
Пусть
case class X(a: Boolean, b: String,c:Int)
case class Y(a: String, b: String)
Определить обобщенное представление Labelled
import shapeless._
import shapeless.ops.product._
import shapeless.syntax.std.product._
object X {
implicit val lgenX = LabelledGeneric[X]
}
object Y {
implicit val lgenY = LabelledGeneric[Y]
}
Определить два класса типов для предоставления методов toMap
object ToMapImplicits {
implicit class ToMapOps[A <: Product](val a: A)
extends AnyVal {
def mkMapAny(implicit toMap: ToMap.Aux[A, Symbol, Any]): Map[String, Any] =
a.toMap[Symbol, Any]
.map { case (k: Symbol, v) => k.name -> v }
}
implicit class ToMapOps2[A <: Product](val a: A)
extends AnyVal {
def mkMapString(implicit toMap: ToMap.Aux[A, Symbol, Any]): Map[String, String] =
a.toMap[Symbol, Any]
.map { case (k: Symbol, v) => k.name -> v.toString }
}
}
Тогда вы можете использовать это так.
object Run extends App {
import ToMapImplicits._
val x: X = X(true, "bike",26)
val y: Y = Y("first", "second")
val anyMapX: Map[String, Any] = x.mkMapAny
val anyMapY: Map[String, Any] = y.mkMapAny
println("anyMapX = " + anyMapX)
println("anyMapY = " + anyMapY)
val stringMapX: Map[String, String] = x.mkMapString
val stringMapY: Map[String, String] = y.mkMapString
println("anyMapX = " + anyMapX)
println("anyMapY = " + anyMapY)
}
который печатает
anyMapX = Карта (c -> 26, b -> велосипед, a -> правда)
anyMapY = Карта (b -> вторая, a -> первая)
stringMapX = Карта (c -> 26, b -> велосипед, a -> правда)
stringMapY = Карта (b -> вторая, a -> первая)
Для вложенных классов дел (таким образом, вложенных карт)
отметьте другой ответ