What is the formal difference in Scala between braces and parentheses, and when should they be used?

后端 未结 9 1144
萌比男神i
萌比男神i 2020-11-22 07:09

What is the formal difference between passing arguments to functions in parentheses () and in braces {}?

The feeling I got from the Pro

相关标签:
9条回答
  • 2020-11-22 07:48

    I don't think there is anything particular or complex about curly braces in Scala. To master the seeming-complex usage of them in Scala, just keep a couple of simple things in mind:

    1. curly braces form a block of code, which evaluates to the last line of code (almost all languages do this)
    2. a function if desired can be generated with the block of code (follows rule 1)
    3. curly braces can be omitted for one-line code except for a case clause (Scala choice)
    4. parentheses can be omitted in function call with code block as a parameter (Scala choice)

    Let's explain a couple of examples per the above three rules:

    val tupleList = List[(String, String)]()
    // doesn't compile, violates case clause requirement
    val filtered = tupleList.takeWhile( case (s1, s2) => s1 == s2 ) 
    // block of code as a partial function and parentheses omission,
    // i.e. tupleList.takeWhile({ case (s1, s2) => s1 == s2 })
    val filtered = tupleList.takeWhile{ case (s1, s2) => s1 == s2 }
    
    // curly braces omission, i.e. List(1, 2, 3).reduceLeft({_+_})
    List(1, 2, 3).reduceLeft(_+_)
    // parentheses omission, i.e. List(1, 2, 3).reduceLeft({_+_})
    List(1, 2, 3).reduceLeft{_+_}
    // not both though it compiles, because meaning totally changes due to precedence
    List(1, 2, 3).reduceLeft _+_ // res1: String => String = <function1>
    
    // curly braces omission, i.e. List(1, 2, 3).foldLeft(0)({_ + _})
    List(1, 2, 3).foldLeft(0)(_ + _)
    // parentheses omission, i.e. List(1, 2, 3).foldLeft(0)({_ + _})
    List(1, 2, 3).foldLeft(0){_ + _}
    // block of code and parentheses omission
    List(1, 2, 3).foldLeft {0} {_ + _}
    // not both though it compiles, because meaning totally changes due to precedence
    List(1, 2, 3).foldLeft(0) _ + _
    // error: ';' expected but integer literal found.
    List(1, 2, 3).foldLeft 0 (_ + _)
    
    def foo(f: Int => Unit) = { println("Entering foo"); f(4) }
    // block of code that just evaluates to a value of a function, and parentheses omission
    // i.e. foo({ println("Hey"); x => println(x) })
    foo { println("Hey"); x => println(x) }
    
    // parentheses omission, i.e. f({x})
    def f(x: Int): Int = f {x}
    // error: missing arguments for method f
    def f(x: Int): Int = f x
    
    0 讨论(0)
  • 2020-11-22 07:51

    There are a couple of different rules and inferences going on here: first of all, Scala infers the braces when a parameter is a function, e.g. in list.map(_ * 2) the braces are inferred, it's just a shorter form of list.map({_ * 2}). Secondly, Scala allows you to skip the parentheses on the last parameter list, if that parameter list has one parameter and it is a function, so list.foldLeft(0)(_ + _) can be written as list.foldLeft(0) { _ + _ } (or list.foldLeft(0)({_ + _}) if you want to be extra explicit).

    However, if you add case you get, as others have mentioned, a partial function instead of a function, and Scala will not infer the braces for partial functions, so list.map(case x => x * 2) won't work, but both list.map({case x => 2 * 2}) and list.map { case x => x * 2 } will.

    0 讨论(0)
  • 2020-11-22 07:53

    Increased compile checking with parens

    The authors of Spray, recommend that round parens give increased compile checking. This is especially important for DSLs like Spray. By using parens you are telling the compiler that it should only be given a single line, therefore if you accidentally gave it two or more, it will complain. Now this isn't the case with curly braces, if for example, you forget an operator somewhere your code will compile, you get unexpected results and potentially a very hard bug to find. Below is contrived (since the expressions are pure and will at least give a warning), but makes the point

    method {
      1 +
      2
      3
    }
    
    method(
      1 +
      2
      3
     )
    

    The first compiles, the second gives error: ')' expected but integer literal found. the author wanted to write 1 + 2 + 3.

    One could argue it's similar for multi-parameter methods with default arguments; it's impossible to accidentally forget a comma to separate parameters when using parens.

    Verbosity

    An important often overlooked note about verbosity. Using curly braces inevitably leads to verbose code since the scala style guide clearly states that closing curly braces must be on their own line: http://docs.scala-lang.org/style/declarations.html "... the closing brace is on its own line immediately following the last line of the function." Many auto-reformatters, like in Intellij, will automatically perform this reformatting for you. So try to stick to using round parens when you can. E.g. List(1, 2, 3).reduceLeft{_ + _} becomes:

    List(1, 2, 3).reduceLeft {
      _ + _
    }
    
    0 讨论(0)
  • 2020-11-22 07:54

    There is an effort from the community to standardize the usage of braces and parentheses, see Scala Style Guide (page 21): http://www.codecommit.com/scala-style-guide.pdf

    The recommended syntax for higher order methods calls is to always use braces, and to skip the dot:

    val filtered = tupleList takeWhile { case (s1, s2) => s1 == s2 }
    

    For "normal" metod calls you should use the dot and parentheses.

    val result = myInstance.foo(5, "Hello")
    
    0 讨论(0)
  • 2020-11-22 08:00

    I think it is worth explaining their usage in function calls and why various things happen. As someone already said curly braces define a block of code, which is also an expression so can be put where expression is expected and it will be evaluated. When evaluated, its statements are executed and last's statement value is the result of whole block evaluation (somewhat like in Ruby).

    Having that we can do things like:

    2 + { 3 }             // res: Int = 5
    val x = { 4 }         // res: x: Int = 4
    List({1},{2},{3})     // res: List[Int] = List(1,2,3)
    

    Last example is just a function call with three parameters, of which each is evaluated first.

    Now to see how it works with function calls let's define simple function that take another function as a parameter.

    def foo(f: Int => Unit) = { println("Entering foo"); f(4) }
    

    To call it, we need to pass function that takes one param of type Int, so we can use function literal and pass it to foo:

    foo( x => println(x) )
    

    Now as said before we can use block of code in place of an expression so let's use it

    foo({ x => println(x) })
    

    What happens here is that code inside {} is evaluated, and the function value is returned as a value of the block evaluation, this value is then passed to foo. This is semantically the same as previous call.

    But we can add something more:

    foo({ println("Hey"); x => println(x) })
    

    Now our code block contains two statements, and because it is evaluated before foo is executed, what happens is that first "Hey" is printed, then our function is passed to foo, "Entering foo" is printed and lastly "4" is printed.

    This looks a bit ugly though and Scala lets us to skip the parenthesis in this case, so we can write:

    foo { println("Hey"); x => println(x) }
    

    or

    foo { x => println(x) }
    

    That looks much nicer and is equivalent to the former ones. Here still block of code is evaluated first and the result of evaluation (which is x => println(x)) is passed as an argument to foo.

    0 讨论(0)
  • 2020-11-22 08:01

    Parenthesis in an ideal coding style is basically used for single line code. But if the particular piece of code is multiline then using braces is a better way.

    0 讨论(0)
提交回复
热议问题