为什么变量newP2Points的计算与newP1Points不同?

时间:2019-11-03 09:32:18

标签: scala probability figaro-lang

我试图理解这段代码背后的想法。该代码详细说明了网球模型,以将各个点建模为集会。

一场网球比赛包括多组比赛,每组包括很多比赛。第一个赢得两盘胜利的玩家将赢得比赛。第一位赢得六场比赛的球员将赢得一套。第一个获得四分的玩家将赢得比赛,除非双方都获得三分。在这种情况下,第一个比对手领先两分的玩家将赢得比赛。

我了解Figaro的功能,但是一些代码行令人困惑。

object TennisGame{
 def tennis( probP1ServeWin: Double, probP1Winner: Double, probP1Error: 
             Double,probP2ServeWin: Double, probP2Winner: Double, 
             probP2Error: Double): Element[Boolean] = {

  def rally(firstShot: Boolean, player1: Boolean): Element[Boolean] = {
     val pWinner = 
      if(firstShot && player1) probP1ServeWin //pWinner will be equal 
                                                //with probP1ServeWinif is 
                                                //the first shot for 
                                                //player1

      else if (firstShot && !player1) probP2ServeWin  
      else if (player1) probP1Winner 
      else probP2Winner
      val pError =if (player1) probP1Error else probP2Error
      val winner = Flip(pWinner)  //winner will be true with a 
                                   //probability equals with pWinner and 
                                  //false with a probability 1- pWinner

      val error = Flip(pError) 

         //With the probability = winner, Constant(player1) is chosen, 
        //producing  player1 with probability 1.0, while with probability
       //1 - winner, If(error, Constant(!player1), rally(false, !player1)) 
      //is chosen

  If(winner, Constant(player1), If(error, Constant(!player1),
  rally(false,!player1)))

    }


 def game(p1Serves: Boolean, p1Points: Element[Int], p2Points: 
          Element[Int]): Element[Boolean] = {

   val p1WinsPoint = rally(true, p1Serves)

    //Apply takes two arguments  p1WinsPoint and p1Points, the Apply’s 
   //function argument takes one Boolean and one integer named wins and 
   //points and calculate the score for player 1
   val newP1Points = Apply(p1WinsPoint, p1Points, (wins: Boolean, points: 
                           Int) => if(wins) points + 1 else points)
   val newP2Points = Apply(p1WinsPoint, p2Points, (wins: Boolean, points: 
                           Int) =>if(wins) points else points + 1)
   val p1WinsGame = Apply(newP1Points, newP2Points, (p1: Int, p2: Int) => 
                          p1 >= 4 && p1- p2 >=2)
   val p2WinsGame = Apply(newP2Points, newP1Points, (p2: Int, p1: Int) => 
                          p2 >= 4 && p2- p1 >=2)
   val gameOver = p1WinsGame || p2WinsGame

  If(gameOver, p1WinsGame, game(p1Serves, newP1Points, newP2Points))
 }

def play(p1Serves: Boolean, p1Sets: Element[Int], p2Sets: Element[Int], 
         p1Games: Element[Int], p2Games: Element[Int]):Element[Boolean] ={

 val p1WinsGame = game( p1Serves, Constant (0), Constant(0))
 val newP1Games = Apply(p1WinsGame, p1Games, p2Games, (wins: Boolean, 
                         p1:Int, p2: Int) =>                                
                         if(wins){
                                  if(p1 >= 5) 0 else p1 + 1 
                            }else
                            {
                                if(p2 >=5) 0 else p1
                            })
val newP2Games = Apply(p1WinsGame, p1Games, p2Games, (wins: Boolean, p1: 
                        Int, p2: Int) =>
                            if(wins){
                               if(p1 >= 5) 0 else p2
                            }else
                            {
                                if(p2 >=5) 0 else p2 + 1
                            })
val newP1Sets = Apply(p1WinsGame, p1Games, p1Sets, (wins: Boolean, games: 
                       Int, sets: Int) =>
                       if(wins && games == 5) sets + 1 else sets)

val newP2Sets = Apply(p1WinsGame, p2Games, p2Sets, (wins: Boolean, games: 
                      Int, sets: Int) =>
                     if(!wins && games == 5) sets + 1 else sets)

val matchOver = Apply(newP1Sets, newP2Sets, (p1: Int, p2: Int) => 
                     p1 >= 2 | p2 >= 2)
 If(matchOver, Apply(newP1Sets, (sets: Int) => sets >= 2), play(!p1Serves, 
    newP1Sets, newP2Sets,newP1Games, newP2Games))
   }   
   play(true, Constant(0), Constant(0), Constant(0), Constant(0)) 
 }

 def main(args: Array[String]){}
}

0 个答案:

没有答案