JohanSJA
JohanSJA

Reputation: 705

Repeated prompt until correct input

I am picking up Scala recently. I have been used to C and Java before. I am wondering if there is a more elegant way of asking for input repeatedly until correct input is given.

val choiceType = {
      var in = ""
      var pass = false
      do {
    in = readLine()
    pass = in match {
        case "1" => println("Not implemented"); true
        case "2" => println("Not implemented"); true
        case "3" => println("Not implemented"); true
        case "4" => println("Not implemented"); true
        case "5" => println("Thanks for using."); true
        case _ => println("Error input. Please enter again. (Possible value: 1 - 5)"); false
    }
      } while (!pass)
      in.toInt
    }
    if (choiceType == 5) System.exit(0)

I am wondering if there is a better way of doing this in Scala?

Upvotes: 3

Views: 1212

Answers (5)

Martin Ring
Martin Ring

Reputation: 5426

import io.Source.stdin

val choices = stdin.getLines.collect {
  case "1" => println("Not implemented")
  case "2" => println("Not implemented")
  case "3" => println("Not implemented")
  case "4" => println("Not implemented")
  case "5" => println("Thanks for using.")
              System.exit(0)
}

choices.next

Upvotes: 2

Daniel C. Sobral
Daniel C. Sobral

Reputation: 297205

Recursion FTW, IMHO. Anyway, I'm gonna suggest a little modification upon Rex Kerr's recursive solution.

def getChoice: String = {
  val line = readLine()
  line match {
    case "1" | "2" | "3" | "4" => println("Not implemented"); line
    case "5" => println("Thanks for using."); line
    case _   => println("Error, blah blah."); getChoice
  }
}

Upvotes: 2

dbyrne
dbyrne

Reputation: 61031

I would use a recursive function. Something like this:

def getChoice: Int = readLine match {
  case x if x < "6" && x > "0" && x.length == 1 => x.toInt
  case _ => println("Error, Possible values: (1 - 5)")
            getChoice
}

Upvotes: 1

David Winslow
David Winslow

Reputation: 8590

Scala allows you to treat { case } blocks as instances of the PartialFunction trait. PartialFunction gives you a way to test whether the function is defined for a particular input. So you could rewrite like this:

val operation: PartialFunction[String, Unit] = {
  case "1" => println("Not implemented")
  case "2" => println("Not implemented")
  case "3" => println("Not implemented")
  case "4" => println("Not implemented")
  case "5" => println("Thanks for using."); System.exit(0)
}

var input: String = ""

do {
  input = readLine()
} while(!operation.isDefinedAt(input))

operation(input)

If you want to avoid having the mutable input variable, you can also use Iterator.continually() (turns an expression into an infinite Iterator which repeatedly evaluates the expression).

val operation = ... // as above
val input = Iterator.continually(readLine()).dropWhile(!operation.isDefinedAt(_)).next
operation(input)

And you can avoid having to give operation a name by using Iterator's collect method.

Iterator.continually(readLine()).collect {
  case "1" => println("one")
}.next

Upvotes: 1

Rex Kerr
Rex Kerr

Reputation: 167901

You could use either Iterate.continually to do the same thing over and over again until you impose some stopping condition (with dropWhile), or you could use Iterator.iterate to give you the previous line in case you want to use it in your error message:

val choiceType = Iterator.iterate(readLine())(line => {
  println("Error input: "+line+".  Please enter again (from 1-5).)")
  readLine()
}).collect(line => line match {
  case "1" => println("Not implemented"); line
  case "2" => println("Not implemented"); line
  case "3" => println("Not implemented"); line
  case "4" => println("Not implemented"); line
  case "5" => println("Thanks for using."); line
}).next.toInt

The way this works is by starting with a readLine, and then if it needs another line it announces an error message based on the previous line (obviously that one was wrong) and reads another line. You then use a collect block to pick out your correct input; the wrong input just falls through without being collected. In this case, since you want to turn it into an integer, I'm just passing the line through. Now, we only want one good entry, so we get the next one and convert it to an int.

You could also use a recursive function to do a similar thing:

def getChoice: String = {
  val line = readLine()
  line match {
    case "1" => println("Not implemented"); line
    case "2" => println("Not implemented"); line
    case "3" => println("Not implemented"); line
    case "4" => println("Not implemented"); line
    case "5" => println("Thanks for using."); line
    case _ => println("Error, blah blah."); getChoice
  }
}
val choiceType = getChoice.toInt

Here the trick is that in the case where you get wrong input, you just call the function again. Since that's the last thing that happens in the function, Scala will avoid a true function call and just jump to the beginning again (tail-recursion), so you won't overflow the stack.

Upvotes: 7

Related Questions