Как преобразовать карту Scala в строку JSON?

Например, у меня есть это значение карты в Scala:

val m = Map(
    "name" -> "john doe", 
    "age" -> 18, 
    "hasChild" -> true, 
    "childs" -> List(
        Map("name" -> "dorothy", "age" -> 5, "hasChild" -> false),
        Map("name" -> "bill", "age" -> 8, "hasChild" -> false)
    )
)

Я хочу преобразовать его в строковое представление JSON:

{
    "name": "john doe",
    "age": 18,
    "hasChild": true,
    "childs": [
        {
            "name": "dorothy",
            "age": 5,
            "hasChild": false
        },
        {
            "name": "bill",
            "age": 8,
            "hasChild": false
        }
    ]
}

Я работаю над Play Framework v2.3, но для решения не нужно использовать библиотеку Play JSON, хотя будет неплохо, если кто-то сможет предоставить как Play, так и non-Play решение.

Это то, что я сделал до сих пор без успеха:

// using jackson library
val mapper = new ObjectMapper()
val res = mapper.writeValueAsString(m)
println(res)

Результат:

{"empty":false,"traversableAgain":true}

Я не понимаю, почему я получил этот результат.

Ответы

Ответ 1

В качестве неиграющего решения вы можете использовать json4s, который предоставляет оболочку вокруг jackson и ее простую в использовании. Если вы используете json4s, вы можете преобразовать карту в json, просто используя:

write(m)                                        
//> res0: String = {"name":"john doe","age":18,"hasChild":true,"childs":[{"name":"dorothy","age":5,"hasChild":false},{"name":"bill","age":8,"hasChild":false}]}

--Updating включить полный example--

import org.json4s._
import org.json4s.native.Serialization._
import org.json4s.native.Serialization
implicit val formats = Serialization.formats(NoTypeHints)

 val m = Map(
  "name" -> "john doe",
  "age" -> 18,
  "hasChild" -> true,
  "childs" -> List(
    Map("name" -> "dorothy", "age" -> 5, "hasChild" -> false),
    Map("name" -> "bill", "age" -> 8, "hasChild" -> false)))

 write(m)

Вывод:

 res0: String = {"name":"john doe","age":18,"hasChild":true,"childs":[{"name" 
 :"dorothy","age":5,"hasChild":false},{"name":"bill","age":8,"hasChild":false }]}

Альтернативный способ:

import org.json4s.native.Json
import org.json4s.DefaultFormats

Json(DefaultFormats).write(m)

Ответ 2

Вам нужно сказать джексону, как бороться с объектами scala: mapper.registerModule(DefaultScalaModule)

Ответ 3

val mapper = new ObjectMapper()
mapper.writeValueAsString(Map("a" -> 1))

result> {"empty": false, "traversableAgain": true}

==============================

import com.fasterxml.jackson.module.scala.DefaultScalaModule

val mapper = new ObjectMapper()
mapper.registerModule(DefaultScalaModule)
mapper.writeValueAsString(Map("a" -> 1))

результат> {"a": 1}

Ответ 4

val mymap = array.map {
  case 1 => ("A", 1)
  case 2 => ("B", 2)
  case 3 => ("C", 3)
}
  .toMap

Используя scala.util.parsing.json.JSONObject, вам нужна только одна строка:

import scala.util.parsing.json.JSONObject

JSONObject(mymap).toString()

Ответ 5

Если вы работаете с четко определенной моделью данных, почему бы не определить классы case и использовать макросы Play JSON для обработки преобразования? т.е.

case class Person(name: String, age: Int, hasChild: Boolean, childs: List[Person])

implicit val fmt = Json.format[Person]

val person = Person(...)

val jsonStr = Json.toJson(person)

Ответ 6

Одна вещь, которую вы можете использовать с помощью библиотеки Jackson, - это использовать объект Java HashMap вместо Scala. Тогда вы можете в основном использовать тот же код "без успеха", который вы уже писали.

import org.codehaus.jackson.map.ObjectMapper
val mapper = new ObjectMapper()
val jmap = new java.util.HashMap[String, Int]()
jmap.put("dog", 4)
jmap.put("cat", 1)
// convert to json formatted string
val jstring  = mapper.writeValueAsString(jmap)
println(jstring)

возвращается

jstring: String = {"dog":4,"cat":1}    

Ответ 7

На случай, если кто-то ищет решение с использованием стандартных библиотек.

def toJson(query: Any): String = query match {
  case m: Map[String, Any] => s"{${m.map(toJson(_)).mkString(",")}}"
  case t: (String, Any) => s""""${t._1}":${toJson(t._2)}"""
  case ss: Seq[Any] => s"""[${ss.map(toJson(_)).mkString(",")}]"""
  case s: String => s""""$s""""
  case null => "null"
  case _ => query.toString
}