Scala:你能在没有括号的表达式中使用"foo match {bar}"吗?

Gui*_*res 9 scala match operator-precedence

为什么这里需要括号?我应该知道一些优先规则吗?

scala> 'x' match { case _ => 1 } + 1
<console>:1: error: ';' expected but identifier found.
       'x' match { case _ => 1 } + 1
                                 ^

scala> ('x' match { case _ => 1 }) + 1
res2: Int = 2
Run Code Online (Sandbox Code Playgroud)

谢谢!

Mat*_*ell 8

正如Agilesteel所说,匹配不被视为简单表达式,也不是if语句,因此您需要用括号括起表达式.从Scala语言规范,6个表达式,p73,匹配是Expr,if是if.+运算符的任何一侧都只接受SimpleExpr.

要将Expr转换为SimpleExpr,必须用()包围它.

复制完整性:

Expr ::= (Bindings | id | ‘_’) ‘=>’ Expr
    | Expr1
Expr1 ::= ‘if’ ‘(’ Expr ‘)’ {nl} Expr [[semi] else Expr]
    | ‘while’ ‘(’ Expr ‘)’ {nl} Expr
    | ‘try’ ‘{’ Block ‘}’ [‘catch’ ‘{’ CaseClauses ‘}’] [‘finally’ Expr]
    | ‘do’ Expr [semi] ‘while’ ‘(’ Expr ’)’
    | ‘for’ (‘(’ Enumerators ‘)’ | ‘{’ Enumerators ‘}’) {nl} [‘yield’] Expr
    | ‘throw’ Expr
    | ‘return’ [Expr]
    | [SimpleExpr ‘.’] id ‘=’ Expr
    | SimpleExpr1 ArgumentExprs ‘=’ Expr
    | PostfixExpr
    | PostfixExpr Ascription
    | PostfixExpr ‘match’ ‘{’ CaseClauses ‘}’
PostfixExpr ::= InfixExpr [id [nl]]
InfixExpr ::= PrefixExpr
    | InfixExpr id [nl] InfixExpr
PrefixExpr ::= [‘-’ | ‘+’ | ‘~’ | ‘!’] SimpleExpr
SimpleExpr ::= ‘new’ (ClassTemplate | TemplateBody)
    | BlockExpr
    | SimpleExpr1 [‘_’]
SimpleExpr1 ::= Literal
    | Path
    | ‘_’
    | ‘(’ [Exprs] ‘)’
    | SimpleExpr ‘.’ id s
    | SimpleExpr TypeArgs
    | SimpleExpr1 ArgumentExprs
    | XmlExpr
Exprs ::= Expr {‘,’ Expr}
BlockExpr ::= ‘{’ CaseClauses ‘}’
    | ‘{’ Block ‘}’
Block ::= {BlockStat semi} [ResultExpr]
ResultExpr ::= Expr1
    | (Bindings | ([‘implicit’] id | ‘_’) ‘:’ CompoundType) ‘=>’ Block
Ascription ::= ‘:’ InfixType
    | ‘:’ Annotation {Annotation}
    | ‘:’ ‘_’ ‘*’
Run Code Online (Sandbox Code Playgroud)


Jai*_*rge 5

经过Scala规范的一些检查后,我想我可以试一试.如果我错了,请纠正我.

首先,一个if或被match定义为Expr- 表达式.

您正在尝试创建中缀表达式(通过在两个表达式之间使用运算符来定义)

但是,规范(第3.2.8节)规定:

fi x运算符中的所有类型都具有相同的优先级; 括号必须用于分组

它还指出:

在fi x操作中的连续类型序列t0 op1 t1 op2...opn tn,所有操作符op1 ,. ..,opn必须具有相同的关联性.如果它们都是左关联的,则序列被解释为(...(t0 op1 t1)op2 ...)opn tn.

所以我的看法是Scala不知道首先要减少什么:匹配或方法'+'调用.

看看这个答案

如果我错了,请纠正我.


agi*_*eel 3

匹配表达式不被视为简单表达式。这是一个类似的例子:

scala> val foo = "bar" + if(3 < 5) 3 else 5   // does not compile
scala> val foo = "bar" + (if(3 < 5) 3 else 5) // does compile
Run Code Online (Sandbox Code Playgroud)

显然你不能在任何你想要的地方编写复杂的表达式。我不知道为什么,希望对此主题有更多了解的人能给您更好的答案。