141 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			141 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
---
 | 
						||
title: Clojure Conditionals
 | 
						||
localeTitle: Clojure Conditionals
 | 
						||
---
 | 
						||
Вы не собираетесь нигде на язык, если все, что вы можете сделать, это определить функции, которые печатают вещи и выполняют простую арифметику. Условные и логические элементы являются фундаментальной частью создания кода, который делает интересные, полезные вещи. Попробуйте представить мир без логики в программах: вы даже не сможете простыми вещами, например, проверить, равны ли два числа!
 | 
						||
 | 
						||
## Логические операторы
 | 
						||
 | 
						||
Clojure, как и большинство языков, имеет 3 логических оператора: `and` , и / `or` `not` . Эти функции принимают логические значения ( `true` или `false` ) в качестве аргументов и возвращают логические значения, основанные на том, что эти логические элементы. Как и все в Lisp, эти операторы используют префиксную нотацию, что означает, что они могут выглядеть странно для вас.
 | 
						||
```
 | 
						||
(and true false) 
 | 
						||
 ; => false 
 | 
						||
 (and true true) 
 | 
						||
 ; => true 
 | 
						||
 (or false false) 
 | 
						||
 ; => false 
 | 
						||
 (or true false) 
 | 
						||
 ; => true 
 | 
						||
 (not true) 
 | 
						||
 ; => false 
 | 
						||
 (not false) 
 | 
						||
 ; => true 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/XfXn8T)
 | 
						||
 | 
						||
## Если
 | 
						||
 | 
						||
`if` позволяет выполнить код на основе того, является ли логическое значение `true` или `false` . `if` в Clojure выглядит довольно странно, не потому, что использует префиксную нотацию, а потому, что ключевого слова else нет. Если условие истинно, оно оценивает первое выражение. Если он ложный, он выполняет второй.
 | 
						||
```
 | 
						||
(if (= (+ 2 2) 4) 
 | 
						||
  (println "Maths works!") ; this gets evaluated if 2 + 2 = 4 
 | 
						||
  (println "UH OH"))       ; this gets evaluated if 2 + 2 != 4 
 | 
						||
 ; => Maths works! 
 | 
						||
 ;    nil 
 | 
						||
```
 | 
						||
 | 
						||
Это создает проблему: что, если мы хотим сделать несколько вещей?
 | 
						||
```
 | 
						||
(if (= (+ 2 2) 4) 
 | 
						||
  (println "Maths works!") 
 | 
						||
  (println "Maths still works!") 
 | 
						||
  (println "UH OH")) 
 | 
						||
 ; => CompilerException java.lang.RuntimeException: Too many arguments to if 
 | 
						||
```
 | 
						||
 | 
						||
К счастью, у нас `do` функция `do` чтобы решить эту проблему. `do` оценку нескольких выражений один за другим.
 | 
						||
```
 | 
						||
(if (= (+ 2 2) 4) 
 | 
						||
  (do                               ; all of this gets evaluated if 2 + 2 = 4 
 | 
						||
    (println "Maths works!") 
 | 
						||
    (println "Maths still works!")) 
 | 
						||
  (println "UH OH")) 
 | 
						||
 ; => Maths works! 
 | 
						||
 ;    Maths still works! 
 | 
						||
 ;    nil 
 | 
						||
```
 | 
						||
 | 
						||
**Примечание:** поскольку `if` есть, само выражение, нет необходимости в тройном операторе, например, во многих C-подобных языках.
 | 
						||
```
 | 
						||
var doesMathsWork = 2 + 2 === 4 ? "Maths works!" : "UH OH"; 
 | 
						||
 console.log(doesMathsWork); 
 | 
						||
 // => Maths works! 
 | 
						||
```
 | 
						||
 | 
						||
Теперь, когда вы видели, как это работает, это выглядит не так странно? Это гораздо легче читать и понимать ( не обращая внимания на отсутствие слова `else` ):
 | 
						||
```
 | 
						||
(def does-maths-work (if (= (+ 2 2) 4) "Maths works!" "UH OH")) 
 | 
						||
 (println does-maths-work) 
 | 
						||
 ; => Maths works! 
 | 
						||
 ;    nil 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/5XhcAa)
 | 
						||
 | 
						||
## Альтернативы if
 | 
						||
 | 
						||
Clojure также имеет некоторые макросы, которые ведут себя аналогично `if` , и иногда могут быть более краткими.
 | 
						||
 | 
						||
`if-not` , пожалуй, самый простой пример - это `if` инвертировать. Эти два фрагмента кода абсолютно одинаковы:
 | 
						||
```
 | 
						||
(def does-maths-work (if (not (= (+ 2 2) 4)) "UH OH" "Maths works!")) 
 | 
						||
 (def does-maths-work (if-not (= (+ 2 2) 4) "UH OH" "Maths works!")) 
 | 
						||
```
 | 
						||
 | 
						||
Первое выражение оценивается, если оно ложно, а второе оценивается, если оно истинно. Обратите внимание на то, что использование `if-not` избегает гнездиться наше состояние внутри `not` , что может помочь облегчить наш код , чтобы понять.
 | 
						||
 | 
						||
`when` это еще один полезный макрос. Эти два фрагмента кода одинаковы:
 | 
						||
```
 | 
						||
(if (= (+ 2 2) 4) (do (println "Maths works!") (println "Hooray!"))) 
 | 
						||
 (when (= (+ 2 2) 4) (println "Maths works!") (println "Hooray!")) 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/tUVAw3)
 | 
						||
 | 
						||
**Примечание:** Нет, `when/else` . `when` выполняется _только в том_ случае, если условие истинно.
 | 
						||
 | 
						||
`cond` позволяет объединить множество условий в одно выражение. Он принимает последовательность пар логического выражения и выражения и оценивает каждое логическое выражение по порядку. Когда он находит логическое выражение, которое оценивает значение `true` , он вычисляет второе выражение пары. После этого никакие другие выражения не оцениваются. Это поведение похоже на логику короткого замыкания в Javascript.
 | 
						||
```
 | 
						||
(cond (= 0 1) "I'm paired with a false expression and I don't evalute.." 
 | 
						||
      (= 1 1) "I'm the first expression paired with a true expression!" 
 | 
						||
      (= 2 2) "I don't evalute even though I'm also paired with true ;_;" 
 | 
						||
      :else   "I evaluate if no other boolean expressions evaluate to true") 
 | 
						||
 ; => "I'm the first expression paired with a true expression!" 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/zu5RCq)
 | 
						||
 | 
						||
Ключевое слово `:else` можно использовать вместо логического выражения в последней паре выражений в `cond` . Это означает, что соответствующее выражение должно быть оценено, если все остальные булевские выражения оцениваются как false. Это то же самое, что `true` в качестве последнего булевского выражения.
 | 
						||
 | 
						||
## Специальные формы и оценка
 | 
						||
 | 
						||
Возможно, вы заметили, что правила оценки условных выражений немного отличаются от других выражений. Условное выражение является частью группы выражений, называемых _специальными формами_ . Это означает, что они не соответствуют нормальным правилам оценки Clojure.
 | 
						||
 | 
						||
Как вы теперь знаете, условное выражение оценивает только подвыражение, которое соответствует логическому результату. Это означает, что неверный код в условном выражении не будет оцениваться в некоторых случаях. Рассмотрим приведенные ниже выражения `if` . Хотя `(+ 1 "failure")` является недопустимым выражением, Clojure генерирует исключение только при условии `false` .
 | 
						||
```
 | 
						||
(if true "sucess" (+ 1 "failure")) 
 | 
						||
 ; => "sucess" 
 | 
						||
 (if false "sucess" (+ 1 "failure")) 
 | 
						||
 ; => ClassCastException java.lang.String cannot be cast to java.lang.Number ... 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/n4Ug2S)
 | 
						||
 | 
						||
Сравните это с поведением `my-if` определенным ниже:
 | 
						||
```
 | 
						||
(defn my-if [condition true-case false-case] 
 | 
						||
  (if condition true-case false-case)) 
 | 
						||
 
 | 
						||
 (my-if true "sucess" (+ 1 "failure")) 
 | 
						||
 ; => ClassCastException java.lang.String cannot be cast to java.lang.Number ... 
 | 
						||
```
 | 
						||
 | 
						||
 [IDEOne!](https://ideone.com/U7cVI4)
 | 
						||
 | 
						||
`my-if` - это функция с нормальными правилами оценки, поэтому все ее подвыражения должны быть опровергнуты до того, как их можно будет оценить.
 | 
						||
 | 
						||
В Clojure есть много полезных макросов, подобных этим для всех видов задач. Попробуйте взглянуть на [документацию Clojure](https://clojuredocs.org/) и посмотреть, сможете ли вы найти ее больше!
 | 
						||
 | 
						||
| [ Предыдущая](//forum.freecodecamp.com/t/clojure-the-basics/18410) | [ Главная ](//forum.freecodecamp.com/t/clojure-resources/18422) | [следующий ](//forum.freecodecamp.com/t/clojure-create-local-variables-with-let/18415) |  
 | 
						||
| [Резюме](//forum.freecodecamp.com/t/clojure-the-basics/18410) | [Содержание](//forum.freecodecamp.com/t/clojure-resources/18422) | [Условные](//forum.freecodecamp.com/t/clojure-create-local-variables-with-let/18415) | |