Clojure: Локальные объявления
def
в Clojure может использоваться как на уровне модуля, так и внутри функций:
(defn f []
(def text "lorem")
(println text))
(f) ; => "lorem"
; у `def text` локальная область видимости
(println text) ; <error>
Но с ним связано несколько тонких моментов:
- Хотя лиспы похожи между собой, конкретно
def
ведет себя совершенно по-разному в разных диалектах Lisp. В некоторых объявления останутся локальными для текущей области видимости, в других же объявление всегда будет глобальным. - Объявления переменных должно идти в самом начале функции, до любых других выражений.
Существует и другой способ объявить локальные переменные, гораздо более популярный и предсказуемый:
(let [text "lorem"] (println text)) ; => lorem
Каждое объявление в let
— это список из имени и выражения, вычисленное значение которого будет ассоциировано с именем. Перепишем уже знакомую нам функцию sum
, используя локальные объявления:
; форма записи без локальных объявлений:
(def sum (fn [x y] (+ x y)))
(sum 8 7) ; вызов глобальной функции
; форма записи с локальными объявлениями:
(let [sum (fn [x y] (+ x y))]
(sum 8 7)) ; вызов локальной функции
Все объявления внутри let
доступны только в выражениях, которые вызываются внутри самого let
после списка объявлений. Вот ещё несколько более сложных примеров, с несколькими объявлениями:
(let [x 2
y (+ 4 3)]
(+ x y)) ; 9
(defn sum-of-squares [x y]
(let [x-square (* x x)
y-square (* y y)]
(+ x-square y-square)))
Можно пойти еще дальше — вызывать локальную функцию внутри глобальной:
(defn sum-of-squares [x y]
(let [square (fn [n] (* n n))]
(+ (square x) (square y))))
(sum-of-squares 8 7) ; => 113
Объявления, созданные в рамках одного использования let
, видны друг другу. Поэтому мы можем сделать так:
(let [x 2
y (* x 10)]
(+ x y))
Есть еще форма letfn
, которая делает то же самое, только для функций, в более коротком виде:
(letfn [(increment [x] (+ x 1))
(decrement [x] (- x 1))
(put-value [] 10)]
(increment (decrement (put-value)))); 10
Задание
Реализуйте функцию prod-sum
, которая сначала умножает переданное число на себя, а затем суммируется между собой и полученным результатом умножения. Воспользуйтесь локальными объявлениями для хранения промежуточных результатов вычисления.
(prod-sum 2) ; 6
(prod-sum 3) ; 12
(prod-sum 4) ; 20
Упражнение не проходит проверку — что делать? 😶
Если вы зашли в тупик, то самое время задать вопрос в «Обсуждениях». Как правильно задать вопрос:
- Обязательно приложите вывод тестов, без него практически невозможно понять что не так, даже если вы покажете свой код. Программисты плохо исполняют код в голове, но по полученной ошибке почти всегда понятно, куда смотреть.
В моей среде код работает, а здесь нет 🤨
Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.
Мой код отличается от решения учителя 🤔
Это нормально 🙆, в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи.
В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.
Прочитал урок — ничего не понятно 🙄
Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в «Обсуждениях». Идеально, если вы сформулируете непонятные моменты в виде вопросов. Обычно нам нужно несколько дней для внесения правок.
Кстати, вы тоже можете участвовать в улучшении курсов: внизу есть ссылка на исходный код уроков, который можно править прямо из браузера.