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
Как с вами связаться? 🙃

Команда проекта находится в телеграм-сообществе по ссылке https://ttttt.me/HexletLearningBot. Там можно задать любой вопрос и повлиять на проект

Упражнение не проходит проверку — что делать? 😶

Если вы зашли в тупик, то самое время поговорить с нашим асситентом Тота во вкладке "Обсуждение". Как правильно задать вопрос:

В моей среде код работает, а здесь нет 🤨

Тесты устроены таким образом, что они проверяют решение разными способами и на разных данных. Часто решение работает с одними входными данными, но не работает с другими. Чтобы разобраться с этим моментом, изучите вкладку «Тесты» и внимательно посмотрите на вывод ошибок, в котором есть подсказки.

Мой код отличается от решения учителя 🤔

Это нормально 🙆, в программировании одну задачу можно выполнить множеством способов. Если ваш код прошел проверку, то он соответствует условиям задачи. В редких случаях бывает, что решение подогнано под тесты, но это видно сразу.

Прочитал урок — ничего не понятно 🙄

Создавать обучающие материалы, понятные для всех без исключения, довольно сложно. Мы очень стараемся, но всегда есть что улучшать. Если вы встретили материал, который вам непонятен, опишите проблему в обратной связи нашего сообщества

Нашли ошибку? Есть что добавить? Пулреквесты приветствуются

Привет! Я Тота и моя задача помочь в обучении. Чтобы активировать меня, нужно зарегистрироваться или залогиниться, если у вас уже есть аккаунт

Loading...

Ваше упражнение проверяется по этим тестам

1(ns define-fun-test
2  (:require [test-helper :refer [assert-solution]]
3            [index :refer [prod-sum]]))
4
5(assert-solution [[1] [2] [3] [4]] [2 6 12 20] prod-sum)
6

Решение учителя откроется через:

20:00
waiting_clock