Подтвердить что ты не робот

Как подождать завершения асинхронных задач в scala?

Я знаю, что моя проблема может показаться немного сложной. Но я постараюсь выразить себя хорошо.

У меня есть этот метод, который я хочу вернуть Map[String, List[String]] с данными.

def myFunction():Map[String, List[String]] = {

  val userMap = Map[String, String](("123456", "ASDBYYBAYGS456789"),
                                    ("54321", "HGFDSA5432"))

  //the result map to return when all data is collected and added
  val resultMap:Future[Map[String, List[String]]]

  //when this map is finished (filled) this map is set to resultMap
  val progressMap = Map[String, List[String]]()

  for(user <- userMap){

     //facebook graph API call to get posts.
     val responsePost = WS.url("async get to facebook url").get()

     responsePosts.flatMap { response => 
        val jsonBody = response.json
        val dataList = List[String]()

        for(i <-0 until 5){

           //parse the json-data to strings
           val messages = (jsonBody.\("statuses").\("data")(i).\("message"))
           val likesArray = (jsonBody.\("statuses").\("data")(i).\\("data")).flatMap(_.as[List[JsObject]])
           val likes = likesArray.length

           //Put post with likes in temporary list
           dataList ::=  ("Post: " + message.toString + " Likes: " + likes.toString)
        }  

           //facebook graph API call to get friends.
           val responseFriends = WS.url("async get to facebook url").get()

           responseFriends.map { response =>
               val jsonBody = response.json
               val friendCount = jsonBody.\("data")(0).\("friend_count").toString

               //add "Friends: xxx" to the dataList and add the new row to resultMap containig a list with post and friends.
               dataList ::= ("Friends: " + friendCount)
               progressMap += user._1 -> dataList

               //check if all users has been updated
               if(progressMap.size == userMap.size){
                  resultMap = progressMap
               }
            }
       }
    }

    //return the resultMap.
    return resultMap
 }
}

Мой код не может быть написан с оптимальным синтаксисом.

Но я хочу вернуть эту resultMap с данными. Моя проблема в том, что, поскольку "get to facebook url" выполняется асинхронно, этот resultMap возвращается пустым. Я не хочу, чтобы это было пустым.

Этот код в моем методе является моим решением до сих пор. Очевидно, это не работает, но я надеюсь, что вы увидите, что я пытаюсь сделать. Не стесняйтесь отвечать своими мыслями, даже если вы не уверены, это может поставить меня на правильный путь.

4b9b3361

Ответ 1

Используйте scala.concurrent.{Future, Promise}:

def doAsyncAction: Promise[T] = {
  val p = Promise[T]
  p success doSomeOperation
  p
}

def useResult = {
   val async = doAsyncAction;
   // The return of the below is Unit.
   async.future onSuccess {
      // do action.
   };
};

Другим способом является результат Await. (это блокирующее действие).

Используется, когда вам нужно вернуть результат

import scala.concurrent.{ ExecutionContext, ExecutionContext$, Future, Promise, Await }
import scala.concurrent.duration._

def method: Option[T] = {
   val future: Future[T] = Future {
       someAction
   }
   val response = future map {
       items => Some(items)
   } recover {
       case timeout: java.util.concurrent.TimeoutException => None
   }
   Await.result(future, 5000 millis);
};

Будьте осторожны при выполнении блокировки фьючерсов в своем собственном исполнителе, иначе вы в конечном итоге блокируете другие параллельные вычисления. Это особенно полезно для запросов S2S и RPC, где блокирование иногда невозможно избежать.