Каждая функция, в своем теле, задает так называемую область видимости. Внутри этой области доступны только те переменные, которые были определены там же. Переменные определённые вне функции, не видимы внутри неё.
<?php
$age = 5;
function generate()
{
print_r($age);
}
generate();
Этот код выдаст предупреждение PHP Notice: Undefined variable: age. Переменная $age
определена вне контекста функции и невидима внутри. Точно такое же поведение и у анонимных функций.
<?php
$age = 5;
$generate = function () {
print_r($age);
};
$generate(); // PHP Notice: Undefined variable: age
Но передача аргументов все же возможна, и делается она благодаря "замыканию". С помощью этого механизма можно "захватить" переменные из внешнего окружения и использовать их внутри. Правда, только для анонимных функций. Термином "замыкание" нередко называют и саму функцию, которая осуществила захват внешних переменных.
<?php
$age = 5;
$generate = function () use ($age) {
print_r($age);
};
$generate(); // 5
Захват переменных осуществляется с помощью ключевого слова use
и списка переменных, который идёт после этого слова. Крайне важно осознать, что use
используется при определении функции, а не её вызове.
Может возникнуть закономерный вопрос, зачем это делать, если переменные можно передать в сам вызов.
<?php
$age = 5;
$generate = function ($age) {
print_r($age);
};
$generate($age); // 5
Замыкания полезны в тех случаях, когда функция определяется в одном месте, а используется в совершенно другом. Замыкание позволяет не таскать за собой гору переменных. А в некоторых ситуациях без них просто не обойтись. Вспомните функцию without()
из пакета Funct
. Эта функция принимает на вход массив и значение, а возвращает новый массив полученный фильтрацией старого по переданному значению. Его реализация, построенная на функциях высшего порядка, подразумевает фильтрацию. Сложность возникает при описании предиката, ведь внутри анонимной функции нужно сравнивать текущее значение и переданный элемент. Замыкание позволяет решить эту задачу просто.
<?php
function without(array $items, $value)
{
$filtered = array_filter($items, function ($item) use ($value) {
return $item !== $value;
});
// Сбрасываем ключи
return array_values($filtered);
}
without([3, 4, 10, 4, 'true'], 4); // [3, 10, 'true']
Без добавления use ($value)
ничего не получится - $value
не виден внутри анонимной функции.
В замыкание можно передать данные по ссылке, что позволит их менять внутри. Но делать так никогда не стоит. Подобное использование создаёт невероятно сложный в отладке и сопровождении код. Единственная причина, по которой можно использовать передачу по ссылке - рекурсия внутри анонимной функции. Подробнее об этом в другом курсе.
Главная цель введения стрелочных функций – создание более лаконичного кода. Поэтому кроме сокращенного синтаксиса, они получили автоматический захват внешних переменных. Другими словами, при использовании стрелочных функций нам не нужен use
.
<?php
$age = 5;
$generate = fn() => print_r($age);
$generate(); // 5
Вам ответят команда поддержки Хекслета или другие студенты.
Выделите текст, нажмите ctrl + enter и отправьте его нам. В течение нескольких дней мы исправим ошибку или улучшим формулировку.
Загляните в раздел «Обсуждение»:
Базовый план откроет полный доступ ко всем курсам, упражнениям и урокам Хекслета, проектам и пожизненный доступ к теории пройденных уроков. Подписку можно отменить в любой момент.
Курсы программирования для новичков и опытных разработчиков. Начните обучение бесплатно.
Наши выпускники работают в компаниях:
С нуля до разработчика. Возвращаем деньги, если не удалось найти работу.
Зарегистрируйтесь или войдите в свой аккаунт