2017-10-11 19 views
1

J'ai une carte qui contient une autre carte dans son champ de valeur. Voici un exemple de certains enregistrements;Extraction de valeurs à partir d'une carte [Chaîne, Any] où Any est une carte elle-même

(8702168053422489,Map(sequence -> 5, id -> 8702168053422489, type -> List(AppExperience, Session), time -> 527780267713)) 
(8702170626376335,Map(trackingInfo -> Map(trackId -> 14183197, location -> Browse, listId -> 3393626f-98e3-4973-8d38-6b2fb17454b5_27331247X28X6839X1506087469573, videoId -> 80161702, rank -> 0, row -> 1, imageKey -> boxshot|AD_e01f4a50-7e2b-11e7-a327-12789459b73e|en, requestId -> 662d92c2-6a1c-41a6-8ac4-bf2ae9f1ce68-417037), id -> 8702170626376335, sequence -> 59, time -> 527780275219, type -> List(NavigationLevel, Session), view -> details)) 
(8702168347359313,Map(muting -> false, id -> 8702168347359313, level -> 1, type -> List(Volume))) 
(8702168321522401,Map(utcOffset -> 3600, type -> List(TimeZone), id -> 8702168321522401)) 
(8702171157207449,Map(trackingInfo -> Map(trackId -> 14183197, location -> Browse, listId -> 3393626f-98e3-4973-8d38-6b2fb17454b5_27331247X28X6839X1506087469573, videoId -> 80161356, rank -> 0, row -> 1, imageKey -> boxshot|AD_e01f4a50-7e2b-11e7-a327-12789459b73e|en, requestId -> 662d92c2-6a1c-41a6-8ac4-bf2ae9f1ce68-417037), id -> 8702171157207449, sequence -> 72, startOffset -> 0, time -> 527780278061, type -> List(StartPlay, Action, Session))) 

Les dossiers réels que j'ai intéressé sont ceux qui contiennent trackingInfo, enregistrements 2 et 5.

Ce que je voudrais faire est d'extraire ceux-ci et puis extraire quelques-unes des clés de là telle comme trackId. Quelque chose comme ça;

val trackingInfo = json("trackingInfo").asInstanceOf[Map[String, Any]] 
val videoId = trackingInfo("videoId").asInstanceOf[Int] 
val id = json("id").asInstanceOf[Long] 
val sequence = json("sequence").asInstanceOf[Int] 
val time = json("time").asInstanceOf[Long] 
val eventType = json.get("type").getOrElse(List("")).asInstanceOf[List[String]] 

Pour extraire la carte intérieure, j'ai fatigué;

Qui ramène la carte interne mais en tant que scala.collection.immutable.Iterable[Any] qui me laisse dans un puzzle pour en extraire des valeurs.

Toute aide est appréciée

+0

Juste un commentaire général. L'utilisation de .asInstanceOf n'est pas sûre. Pourrait causer des problèmes – Pavel

Répondre

1

Disons que vous avez une vraie carte (je l'ai coupé un peu)

val data: Map[ BigInt, Any ] = Map(
    BigInt(8702168053422489L) -> Map("sequence" -> "5", "id" -> BigInt(8702168053422489L)), 
    BigInt(8702170626376335L) -> Map("trackingInfo" -> Map("trackId" -> BigInt(14183197), "location" -> "Browse"), "id" -> BigInt(8702170626376335L)), 
    BigInt(8702168347359313L) -> Map("id" -> BigInt(8702168347359313L)), 
    BigInt(8702168321522401L) -> Map("id" -> BigInt(8702168321522401L)), 
    BigInt(8702171157207449L) -> Map("trackingInfo" -> Map("trackId" -> BigInt(14183197), "location" -> "Browse"), "id" -> BigInt(8702171157207449L)) 
) 

Et vous voulez obtenir des enregistrements qui ont une trackingInfo clé

val onlyWithTracking = data.filter((row) => { 

    val recordToFilter = row._2 match { 
     case trackRecord: Map[ String, Any ] => trackRecord 
     case _ => Map("trackId" -> Map()) 
    } 

    recordToFilter.contains("trackingInfo") 
}) 

Puis traiter ces enregistrements d'une manière ou d'une autre

onlyWithTracking.foreach((row) => { 

    val record = row._2 match { 
     case trackRecord: Map[ String, Any ] => trackRecord 
     case _ => Map("trackingInfo" -> Map()) 
    } 

    val trackingInfo = record("trackingInfo") match { 
     case trackRow: Map[ String, Any ] => trackRow 
     case _ => Map("trackId" -> "error") 
    } 

    val trackId = trackingInfo("trackId") 

    println(trackId) 
}) 

Avec ce modèle de correspondance, j'essaie de m'assurer que l'utilisation de touches comme trackingInfo ou trackId est relativement sûre. Vous devriez mettre en œuvre une approche plus stricte.

+0

merci pour votre réponse. cependant rappelez-vous que la carte originale est 'Map [String, Any]' alors quand vous faites 'val onlyWithTracking = data.filter ((row) => row._2.contains (" trackingInfo ")) la valeur' Je reçois l'erreur 'contient n'est pas membre de Any' – ukbaz

+0

Ouais. J'ai édité la réponse pour répondre à cela. Un peu plus de correspondance de motif. –

+0

merci pour cela, je semble obtenir cette erreur sur le bit de traitement 'scala.math.BigInt ne peut pas être converti en java.lang.String'. – ukbaz