Я однажды пытался написать об этом, но в конце концов сдался, так как правила несколько размыты. По сути, вам придется освоить его.
Возможно, лучше сконцентрироваться на том, где фигурные скобки и скобки можно использовать взаимозаменяемо: при передаче параметров в вызовы методов. Вы можете заменить круглые скобки круглыми скобками, если и только если метод ожидает один параметр. Например:
List(1, 2, 3).reduceLeft{_ + _} // valid, single Function2[Int,Int] parameter
List{1, 2, 3}.reduceLeft(_ + _) // invalid, A* vararg parameter
Однако вам нужно знать больше, чтобы лучше понять эти правила.
Увеличена проверка компиляции с помощью паренов
Авторы Spray рекомендуют использовать круглые скобки, потому что они повышают проверку компиляции. Это особенно важно для DSL, таких как Spray. Используя parens, вы говорите компилятору, что он должен содержать только одну строку; поэтому, если вы случайно дадите ему два или более, он будет жаловаться. Теперь это не относится к фигурным скобкам - если, например, вы где-то забудете оператор, ваш код скомпилируется, и вы получите неожиданные результаты и, возможно, очень сложную ошибку для поиска. Ниже выдумано (поскольку выражения являются чистыми и, по крайней мере, даст предупреждение), но подчеркивает:
method {
1 +
2
3
}
method(
1 +
2
3
)
Первый компилирует, второй дает error: ')' expected but integer literal found
. Автор хотел написать 1 + 2 + 3
.
Можно утверждать, что это похоже на многопараметрические методы с аргументами по умолчанию; невозможно случайно забыть запятую для разделения параметров при использовании паренов.
многословие
Важное часто пропускаемое примечание о многословии. Использование фигурных скобок неизбежно приводит к многословному коду, поскольку в руководстве по стилю Scala четко указано, что закрывающие фигурные скобки должны быть на отдельной строке:
… Закрывающая скобка находится на отдельной строке сразу после последней строки функции.
Многие автоформаты, как в IntelliJ, автоматически выполнят это переформатирование для вас. Поэтому старайтесь использовать круглые скобки, когда можете.
Инфиксная нотация
При использовании инфиксной нотации, например, List(1,2,3) indexOf (2)
вы можете опустить круглые скобки, если есть только один параметр, и записать его как List(1, 2, 3) indexOf 2
. Это не случай точечной нотации.
Также обратите внимание, что когда у вас есть один параметр, который является выражением с несколькими токенами, например x + 2
или a => a % 2 == 0
, вы должны использовать круглые скобки, чтобы указать границы выражения.
Кортеж
Поскольку иногда можно опустить круглые скобки, иногда кортежу требуется дополнительная скобка, например, в ((1, 2))
, а иногда внешняя скобка может быть опущена, например, в (1, 2)
. Это может вызвать путаницу.
Литералы функций / частичных функций с case
Scala имеет синтаксис для литералов функций и частичных функций. Это выглядит так:
{
case pattern if guard => statements
case pattern => statements
}
Единственный другие местами , где вы можете использовать case
оператор являются с match
и catch
ключевыми словами:
object match {
case pattern if guard => statements
case pattern => statements
}
try {
block
} catch {
case pattern if guard => statements
case pattern => statements
} finally {
block
}
Вы не можете использовать case
заявления в любом другом контексте . Итак, если вы хотите использовать case
, вам нужны фигурные скобки. Если вам интересно, что делает различие между функцией и частичной функцией буквальным, ответ: контекст. Если Scala ожидает функцию, функция, которую вы получите. Если он ожидает частичную функцию, вы получите частичную функцию. Если оба ожидаются, это дает ошибку о неоднозначности.
Выражения и блоки
Скобки могут быть использованы для создания подвыражений. Фигурные скобки могут использоваться для создания блоков кода (это не буквальная функция, поэтому остерегайтесь использовать ее как единое целое). Блок кода состоит из нескольких операторов, каждый из которых может быть оператором импорта, объявлением или выражением. Это выглядит так:
{
import stuff._
statement ; // ; optional at the end of the line
statement ; statement // not optional here
var x = 0 // declaration
while (x < 10) { x += 1 } // stuff
(x % 5) + 1 // expression
}
( expression )
Итак, если вам нужны объявления, несколько операторов import
или что-то в этом роде, вам нужны фигурные скобки. А поскольку выражение является утверждением, скобки могут появляться внутри фигурных скобок. Но самое интересное заключается в том , что блоки кода являются также выражения, так что вы можете использовать их в любом месте внутри выражения:
( { var x = 0; while (x < 10) { x += 1}; x } % 5) + 1
Таким образом, поскольку выражения являются операторами, а блоки кодов являются выражениями, все приведенное ниже является действительным:
1 // literal
(1) // expression
{1} // block of code
({1}) // expression with a block of code
{(1)} // block of code with an expression
({(1)}) // you get the drift...
Где они не взаимозаменяемы
В принципе, вы не можете заменить {}
с ()
или наоборот где - нибудь еще. Например:
while (x < 10) { x += 1 }
Это не вызов метода, поэтому вы не можете написать его другим способом. Ну, вы можете поставить фигурные скобки внутри скобок для condition
, а также использовать скобки внутри фигурных скобок для блока кода:
while ({x < 10}) { (x += 1) }
Надеюсь, это поможет.