Зарегистрируйтесь, чтобы продолжить обучение

Лямбда-функции Java: Функции

До сих пор в Java мы встречались только с методами, хотя и слышали термин «функция». В отличие от методов, функции существуют сами по себе без привязки к классу. Технически Java не позволяет создавать подобные функции, поэтому почти всегда, когда нужна обычная функция, в Java создается статический метод. Пример:

public class MathUtils {
    // Статический метод для вычисления факториала числа
    public static int factorial(int number) {
        if (number < 0) {
            throw new IllegalArgumentException("Number must be non-negative.");
        }

        int result = 1;
        for (int i = 2; i <= number; i++) {
            result *= i;
        }
        return result;
    }

    public static void main(String[] args) {
        int number = 5;
        // Вызов статического метода
        int result = MathUtils.factorial(number);
        System.out.println("Factorial of " + number + " is: " + result);
    }
}

В таком смысле, статические методы вполне заменяют обычные функции, хотя и делают код более многословным. Но встречаются и другие ситуации, где функции удобны и речь здесь идет про лямбда-функции или, как их еще называют, анонимные функции.

forEach() в Map

Рассмотрим на примере Map. Для того чтобы обойти пары ключ-значения, мы скорее всего воспользуемся методом entrySet(), который возвращает специальный объект, содержащий и ключ и значение.

var codes = new HashMap<String, Integer>();
codes.put("usa", 1);
codes.put("france", 33);
codes.put("germany", 49);
var entries = codes.entrySet();
for (var entry : entries) {
    System.out.println(entry.getKey());
    System.out.println(entry.getValue());
}

Но эту же задачу можно решить проще, если воспользоваться методом forEach(), который работает с лямбда-функциями.

codes.forEach((key, value) -> {
    System.out.println(key);
    System.out.println(value);
});

В этом примере мы используем метод forEach(), параметром которого является лямбда-функция. Эта лямбда-функция принимает на вход два параметра: ключ и значение. В ее теле выполняется тот код, который мы напишем. Вызов этой функции происходит внутри метода forEach() для каждой пары ключ-значение. Вот как выглядит лямбда-функция сама по себе, без forEach():

(key, value) -> {
    System.out.println(key);
    System.out.println(value);
}

Лямбда-функция записывается так: () -> {}. То, что в скобках – это параметры. То, что в фигурных скобках – это тело функции. Знак -> отделяет параметры от тела. Особенностью лямбда-функций является то, что параметры указываются без типов данных. Мы разберем это подробнее, когда будем учиться создавать методы, работающие с лямбда-функциями.

forEach() в списках

Похожая реализация forEach() есть и в списках. Здесь лямбда-функция принимает на вход один параметр.

var temperatures = List.of(20, 25, 18, 19, 22, 17, 20);
temperatures.forEach((value) -> {
    System.out.println(value);
});

Пример выше можно переписать в другом, более компактном синтаксисе:

temperatures.forEach(value -> System.out.println(value));

В этом примере произошло два упрощения:

  • Убраны фигурные скобки у тела функции — такое допустимо, если внутри тела выполняется одна инструкция
  • Убраны скобки вокруг параметра — это допустимо, если параметр всего один

Упрощенный синтаксис лямбда-функций

Лямбда-функции могут не только выполнять какое-то действие, но и возвращать значение. Рассмотрим на примере метода replaceAll() в списках. Метод заменяет каждый элемент списка на результат, который вернет лямбда-функция для текущего элемента

var temperatures = new ArrayList<Integer>();
temperatures.addAll(List.of(-2, 5, 2));

temperatures.replaceAll(value -> {
    return value + 273;
});

System.out.println(temperatures); // => [271, 278, 275]

В Java существует альтернативный вариант записи лямбда-функций

temperatures.replaceAll(value -> value + 273);

В этом примере мы опустили фигурные скобки у тела функции и ключевое слово return. Так можно делать, когда лямбда-функция содержит только одно выражение, результат которого должен быть возвращен. В этом случае Java автоматически возвращает результат выражения без явного использования return

Ограничение на использование контекста

Как и в случае циклов, мы можем внутри тела лямбда-функции использовать данные взятые из внешнего контекста, то есть определенные вне тела функции.

var temperatures = List.of(20, 25, 18, 19, 22, 17, 20);
var n = 2;
temperatures.forEach((value) -> {
    System.out.println(value * n);
});

Но есть ограничение. В отличие от циклов, где с этими данными можно выполнять любые действия, лямбда-функции ограничены тем, что не могут изменять значение переменных, с которыми идет работа:

var temperatures = List.of(20, 25, 18, 19, 22, 17, 20);
var n = 10;
temperatures.forEach((value) -> {
    System.out.println(value * n--);
});

Подобный код приведет к ошибке: Local variable n defined in an enclosing scope must be final or effectively final


Дополнительные материалы

  1. Метод forEach() в списках
  2. Метод forEach() в Map

Аватары экспертов Хекслета

Остались вопросы? Задайте их в разделе «Обсуждение»

Вам ответят команда поддержки Хекслета или другие студенты

Для полного доступа к курсу нужен базовый план

Базовый план откроет полный доступ ко всем курсам, упражнениям и урокам Хекслета, проектам и пожизненный доступ к теории пройденных уроков. Подписку можно отменить в любой момент.

Получить доступ
1000
упражнений
2000+
часов теории
3200
тестов

Открыть доступ

Курсы программирования для новичков и опытных разработчиков. Начните обучение бесплатно

  • 130 курсов, 2000+ часов теории
  • 1000 практических заданий в браузере
  • 360 000 студентов
Отправляя форму, вы принимаете «Соглашение об обработке персональных данных» и условия «Оферты», а также соглашаетесь с «Условиями использования»

Наши выпускники работают в компаниях:

Логотип компании Альфа Банк
Логотип компании Aviasales
Логотип компании Yandex
Логотип компании Tinkoff
Рекомендуемые программы
профессия
Программирование на Java, Разработка веб-приложений и микросервисов используя Spring Boot, проектирование REST API
10 месяцев
с нуля
Старт 23 января

Используйте Хекслет по-максимуму!

  • Задавайте вопросы по уроку
  • Проверяйте знания в квизах
  • Проходите практику прямо в браузере
  • Отслеживайте свой прогресс

Зарегистрируйтесь или войдите в свой аккаунт

Отправляя форму, вы принимаете «Соглашение об обработке персональных данных» и условия «Оферты», а также соглашаетесь с «Условиями использования»
Изображение Тото

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