Конфигурационные API: Provider, Property, ListProperty
Gradle предоставляет API для ленивой конфигурации, что улучшает производительность и гибкость.
Provider:
Интерфейс для ленивых значений, вычисляемых при необходимости.
Пример:
Property:
Для управления одиночными значениями.
Пример:
ListProperty:
Для управления списками значений.
Пример:
Kotlin DSL:
Нюансы:
Используйте Provider для динамических вычислений.
Property и ListProperty обеспечивают строгую типизацию, особенно в Kotlin DSL.
Ext-переменные (ext {})
Блок ext позволяет определять пользовательские свойства для проекта или объектов.
Пример:
Kotlin DSL:
Использование:
Для хранения общих констант (например, версий зависимостей).
Для передачи данных между задачами или модулями.
Нюансы:
Избегайте чрезмерного использования ext в пользу Version Catalog для версий зависимостей.
Используйте для кастомных настроек, не связанных с зависимостями.
buildSrc/ директория
Директория buildSrc позволяет определять кастомную логику (плагины, задачи) в проекте.
Структура:
Пример плагина (buildSrc/src/main/groovy/com/example/MyPlugin.groovy):
Настройка (buildSrc/build.gradle):
Использование:
Kotlin DSL:
Создайте buildSrc/src/main/kotlin/com/example/MyPlugin.kt:
Настройте buildSrc/build.gradle.kts:
Нюансы:
buildSrc автоматически доступен всем модулям проекта.
Используйте для кастомных задач и плагинов, специфичных для проекта.
Для повторно используемых плагинов предпочтительнее создавать отдельный проект и публиковать в репозиторий.
#Java #middle #Gradle #Task #deploy
Gradle предоставляет API для ленивой конфигурации, что улучшает производительность и гибкость.
Provider:
Интерфейс для ленивых значений, вычисляемых при необходимости.
Пример:
def versionProvider = providers.provider { project.version }
task printVersion {
doLast {
println "Version: ${versionProvider.get()}"
}
}
Property:
Для управления одиночными значениями.
Пример:
task example {
def outputFile = objects.property(String)
outputFile.set('build/output.txt')
doLast {
println "Output: ${outputFile.get()}"
}
}
ListProperty:
Для управления списками значений.
Пример:
task example {
def files = objects.listProperty(String)
files.set(['file1.txt', 'file2.txt'])
doLast {
files.get().each { println it }
}
}
Kotlin DSL:
tasks.register("example") {
val outputFile = objects.property<String>()
outputFile.set("build/output.txt")
doLast {
println("Output: ${outputFile.get()}")
}
}
В памяти: Provider, Property и ListProperty хранят ссылки на значения, а не сами значения, минимизируя потребление памяти до фазы выполнения. Это снижает overhead на фазе конфигурации (экономия 10-50 МБ на задачу).
Нюансы:
Используйте Provider для динамических вычислений.
Property и ListProperty обеспечивают строгую типизацию, особенно в Kotlin DSL.
Ext-переменные (ext {})
Блок ext позволяет определять пользовательские свойства для проекта или объектов.
Пример:
ext {
myVersion = '1.0.0'
myProperty = 'value'
}
task printExt {
doLast {
println project.ext.myVersion
}
}
Kotlin DSL:
extra["myVersion"] = "1.0.0"
extra["myProperty"] = "value"
tasks.register("printExt") {
doLast {
println(project.extra["myVersion"])
}
}
Использование:
Для хранения общих констант (например, версий зависимостей).
Для передачи данных между задачами или модулями.
В памяти: ext-переменные хранятся как часть модели проекта, добавляя минимальный overhead (менее 10 МБ). Однако большое количество переменных может усложнить модель.
Нюансы:
Избегайте чрезмерного использования ext в пользу Version Catalog для версий зависимостей.
Используйте для кастомных настроек, не связанных с зависимостями.
buildSrc/ директория
Директория buildSrc позволяет определять кастомную логику (плагины, задачи) в проекте.
Структура:
my-project/
├── buildSrc/
│ ├── src/main/groovy/com/example/MyPlugin.groovy
│ ├── build.gradle
├── build.gradle
├── settings.gradle
Пример плагина (buildSrc/src/main/groovy/com/example/MyPlugin.groovy):
package com.example
import org.gradle.api.Plugin
import org.gradle.api.Project
class MyPlugin implements Plugin<Project> {
void apply(Project project) {
project.tasks.register('myTask') {
doLast {
println 'Hello from buildSrc plugin!'
}
}
}
}
Настройка (buildSrc/build.gradle):
plugins {
id 'groovy'
}
Использование:
apply plugin: 'com.example.my-plugin'
Kotlin DSL:
Создайте buildSrc/src/main/kotlin/com/example/MyPlugin.kt:
package com.example
import org.gradle.api.Plugin
import org.gradle.api.Project
class MyPlugin : Plugin<Project> {
override fun apply(project: Project) {
project.tasks.register("myTask") {
doLast {
println("Hello from buildSrc plugin!")
}
}
}
}
Настройте buildSrc/build.gradle.kts:
plugins {
`kotlin-dsl`
}
В памяти: buildSrc компилируется как отдельный проект, добавляя собственный classpath и зависимости в JVM. Это увеличивает потребление памяти (100-200 МБ), особенно если buildSrc включает сложные плагины.
Нюансы:
buildSrc автоматически доступен всем модулям проекта.
Используйте для кастомных задач и плагинов, специфичных для проекта.
Для повторно используемых плагинов предпочтительнее создавать отдельный проект и публиковать в репозиторий.
#Java #middle #Gradle #Task #deploy
👍1
Shared Build Logic через init-скрипты
Init-скрипты позволяют задавать глобальную логику, применяемую ко всем Gradle-сборкам.
Настройка:
Создайте файл ~/.gradle/init.d/my-init.gradle:
Kotlin DSL (~/.gradle/init.d/my-init.gradle.kts):
Использование:
Init-скрипты автоматически применяются ко всем проектам.
Можно указать скрипт явно:
Нюансы:
Используйте для глобальных настроек (например, репозитории, JVM-параметры).
Избегайте сложной логики, чтобы не замедлять инициализацию.
Gradle Wrapper (gradlew): настройка и версионирование
Gradle Wrapper (gradlew) — это скрипт, обеспечивающий воспроизводимость сборки с фиксированной версией Gradle.
Настройка:
Сгенерируйте Wrapper:
Создает файлы:
gradle-wrapper.properties:
Использование:
Запускайте сборку через Wrapper:
Wrapper загружает указанную версию Gradle, если она отсутствует в ~/.gradle/wrapper.
Версионирование:
Обновите версию:
Используйте конкретную версию для согласованности в CI/CD.
Нюансы:
Всегда включайте Wrapper в репозиторий для воспроизводимости.
Проверяйте целостность distributionUrl для безопасности (используйте HTTPS).
Настройте прокси в gradle-wrapper.properties, если требуется:systemProp.http.proxyHost=proxy.example.com
systemProp.http.proxyPort=8080
#Java #middle #Gradle #Task #deploy
Init-скрипты позволяют задавать глобальную логику, применяемую ко всем Gradle-сборкам.
Настройка:
Создайте файл ~/.gradle/init.d/my-init.gradle:
allprojects {
repositories {
mavenCentral()
}
}
gradle.projectsLoaded {
rootProject.tasks.register('globalTask') {
doLast {
println 'Global task from init script'
}
}
}
Kotlin DSL (~/.gradle/init.d/my-init.gradle.kts):
allprojects {
repositories {
mavenCentral()
}
}
gradle.projectsLoaded {
rootProject.tasks.register("globalTask") {
doLast {
println("Global task from init script")
}
}
}
Использование:
Init-скрипты автоматически применяются ко всем проектам.
Можно указать скрипт явно:
./gradlew build --init-script my-init.gradle
В памяти: Init-скрипты загружаются на фазе инициализации, добавляя задачи и конфигурации в модель проекта. Это увеличивает overhead (10-50 МБ), но распределяется между всеми сборками.
Нюансы:
Используйте для глобальных настроек (например, репозитории, JVM-параметры).
Избегайте сложной логики, чтобы не замедлять инициализацию.
Gradle Wrapper (gradlew): настройка и версионирование
Gradle Wrapper (gradlew) — это скрипт, обеспечивающий воспроизводимость сборки с фиксированной версией Gradle.
Настройка:
Сгенерируйте Wrapper:
gradle wrapper --gradle-version 8.1
Создает файлы:
my-project/
├── gradlew
├── gradlew.bat
├── gradle/
│ └── wrapper/
│ ├── gradle-wrapper.jar
│ ├── gradle-wrapper.properties
gradle-wrapper.properties:
distributionUrl=https\://services.gradle.org/distributions/gradle-8.1-bin.zip
Использование:
Запускайте сборку через Wrapper:
./gradlew build
Wrapper загружает указанную версию Gradle, если она отсутствует в ~/.gradle/wrapper.
Версионирование:
Обновите версию:
./gradlew wrapper --gradle-version 8.2
Используйте конкретную версию для согласованности в CI/CD.
В памяти: Wrapper загружает Gradle в отдельный JVM-процесс, добавляя базовый overhead (200-300 МБ). Кэширование дистрибутива в ~/.gradle/wrapper минимизирует сетевые запросы.
Нюансы:
Всегда включайте Wrapper в репозиторий для воспроизводимости.
Проверяйте целостность distributionUrl для безопасности (используйте HTTPS).
Настройте прокси в gradle-wrapper.properties, если требуется:systemProp.http.proxyHost=proxy.example.com
systemProp.http.proxyPort=8080
#Java #middle #Gradle #Task #deploy
👍1
Как Вы относитесь к "накрутке" опыта в резюме?
Anonymous Poll
18%
Положительно! Хочу больше деняк! 😎
18%
Положительно, ибо нефиг HR искать только Senior-ов 😏
15%
Безразлично. Я этим не пользуюсь. 🖕
27%
Негативно, но больше вынужденно, из-за отсутствия опыта 🤷♀️
21%
Негативно, накрутчики рушат рынок 😈
👍2
Тестирование в Spring
1. Простые юнит тесты через TDD в Spring
Серия видео посвященная тестированию всего и вся в Spring.
Что мы рассмотрели:
🔵 Что такое TDD, его принципы и как по нему работать в java
🔵 Рассмотрели простое юнит создание тестов, как с поднятием контекста, так и без.
🔵 Узнали для чего те или иные аннотации без которых не получится написать тесты.
Написали минимальный Spring-сервис через TDD, написав сначала тесты.
Ссылка на Youtube
Ссылка на Рутьюб
Ссылка на GitHub - жду ваших звезд☺️
Смотрите, ставьте лайки, подписывайтесь на каналы!✌️
1. Простые юнит тесты через TDD в Spring
Серия видео посвященная тестированию всего и вся в Spring.
Что мы рассмотрели:
Написали минимальный Spring-сервис через TDD, написав сначала тесты.
Ссылка на Youtube
Ссылка на Рутьюб
Ссылка на GitHub - жду ваших звезд
Смотрите, ставьте лайки, подписывайтесь на каналы!
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2🔥2
Что выведет код?
#Tasks
public class Task060825 {
public static void main(String[] args) {
int x = 1;
int y = 2;
if (++x > y++ ? x++ < --y : y-- > ++x) {
System.out.println("A: x=" + x + " y=" + y);
} else {
System.out.println("B: x=" + x + " y=" + y);
}
}
}
#Tasks
🤯2👍1
👍3
Что такое SoftReference и когда ее использовать? 🤓
Ответ:
SoftReference — это ссылка, которая сохраняется сборщиком мусора до тех пор, пока память не станет критически низкой. Подходит для кэшей, где данные можно восстановить.
Пример:
SoftReference<String> softRef = new SoftReference<>(new String("cache"));
System.out.println(softRef.get()); // cache
System.gc(); // Может не удалить, если память не нужна
В отличие от WeakReference, SoftReference живет дольше, что делает ее предпочтительной для кэширования.
#собеседование
Ответ:
Пример:
SoftReference<String> softRef = new SoftReference<>(new String("cache"));
System.out.println(softRef.get()); // cache
System.gc(); // Может не удалить, если память не нужна
В отличие от WeakReference, SoftReference живет дольше, что делает ее предпочтительной для кэширования.
#собеседование
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
Джи́мми До́нал Уэ́йлс (англ. Jimmy Donal Wales, также известный под псевдонимом Джи́мбо (англ. Jimbo); род. 7 августа 1966, Хантсвилл, Алабама, США) — американский и британский интернет-предприниматель, один из создателей концепции «вики», основатель онлайн-энциклопедии Википедии.
1944 — Презентация IBM Mark I (ASCC)
Первый крупный механический программируемый компьютер, использовался для инженерных расчётов во Второй мировой.
1955 — Первый транзисторный радиоприёмник (Sony TR‑55)
Появление портативной электроники и начало эры миниатюрных устройств.
1959 — Запуск спутника Explorer 6 (США)
Первое в истории изображение Земли, переданное из космоса.
1981 — IBM представила первый ПК (IBM 5150)
Начало массовой компьютеризации, предтеча всех современных персональных компьютеров.
1991 — Публикация World Wide Web
Тим Бернерс‑Ли сделал первую веб‑страницу доступной в сети. Рождение современного Интернета.
#Biography #Birth_Date #Events #7Августа
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2
Switch/case в Java
Оператор switch в Java позволяет выбирать, какой код выполнять, в зависимости от значения переменной. Он часто используется вместо множества if-else условий, когда нужно проверить одно значение на совпадение с несколькими вариантами. С Java 14 switch также поддерживает switch expressions (выражения), которые делают код короче и удобнее.
1. Что такое switch в Java?
switch — это условный оператор, который проверяет значение переменной и выполняет соответствующий блок кода в зависимости от совпадения с заданными вариантами (case). Switch expressions — это улучшенная версия switch, которая позволяет возвращать значение и писать более компактный код.
Зачем нужен switch?
Простота выбора: Удобен, когда нужно выбрать действие из нескольких вариантов.
Читаемость: Делает код чище, чем длинная цепочка if-else.
Гибкость: Поддерживает разные типы данных и новые возможности (switch expressions).
Эффективность: В некоторых случаях работает быстрее, чем if-else.
2. Синтаксис switch
2.1. Классический switch (оператор)
Классический switch выполняет блок кода для совпадающего case и использует break, чтобы выйти из конструкции.
Синтаксис:
Пример:
2.2. Switch Expression (с Java 14)
Switch expressions позволяют возвращать значение и использовать стрелочный синтаксис (->) для упрощения кода. Они не требуют break, так как автоматически завершают выполнение.
Синтаксис:
Пример:
Примечания к синтаксису:
Поддерживаемые типы:
Классический switch: byte, short, int, char, String, enum.
Switch expression: Те же типы + возможность возвращать значения.
break: В классическом switch нужен, чтобы избежать "проваливания" (fall-through) в следующий case.
default: Выполняется, если ни один case не совпал. Необязателен, но рекомендуется.
Стрелочный синтаксис (->): Используется в switch expressions для компактности.
#Java #для_новичков #beginner #switch_case
Оператор switch в Java позволяет выбирать, какой код выполнять, в зависимости от значения переменной. Он часто используется вместо множества if-else условий, когда нужно проверить одно значение на совпадение с несколькими вариантами. С Java 14 switch также поддерживает switch expressions (выражения), которые делают код короче и удобнее.
1. Что такое switch в Java?
switch — это условный оператор, который проверяет значение переменной и выполняет соответствующий блок кода в зависимости от совпадения с заданными вариантами (case). Switch expressions — это улучшенная версия switch, которая позволяет возвращать значение и писать более компактный код.
Зачем нужен switch?
Простота выбора: Удобен, когда нужно выбрать действие из нескольких вариантов.
Читаемость: Делает код чище, чем длинная цепочка if-else.
Гибкость: Поддерживает разные типы данных и новые возможности (switch expressions).
Эффективность: В некоторых случаях работает быстрее, чем if-else.
2. Синтаксис switch
2.1. Классический switch (оператор)
Классический switch выполняет блок кода для совпадающего case и использует break, чтобы выйти из конструкции.
Синтаксис:
switch (выражение) {
case значение1:
// Код для значения1
break;
case значение2:
// Код для значения2
break;
default:
// Код, если ни один case не совпал
}
Пример:
int day = 3;
switch (day) {
case 1:
System.out.println("Понедельник");
break;
case 2:
System.out.println("Вторник");
break;
case 3:
System.out.println("Среда");
break;
default:
System.out.println("Другой день");
}
Вывод: Среда
2.2. Switch Expression (с Java 14)
Switch expressions позволяют возвращать значение и использовать стрелочный синтаксис (->) для упрощения кода. Они не требуют break, так как автоматически завершают выполнение.
Синтаксис:
switch (выражение) {
case значение1 -> результат1;
case значение2 -> результат2;
default -> результат_по_умолчанию;
}
Пример:
int day = 3;
String dayName = switch (day) {
case 1 -> "Понедельник";
case 2 -> "Вторник";
case 3 -> "Среда";
default -> "Другой день";
};
System.out.println(dayName);
Вывод: Среда
Примечания к синтаксису:
Поддерживаемые типы:
Классический switch: byte, short, int, char, String, enum.
Switch expression: Те же типы + возможность возвращать значения.
break: В классическом switch нужен, чтобы избежать "проваливания" (fall-through) в следующий case.
default: Выполняется, если ни один case не совпал. Необязателен, но рекомендуется.
Стрелочный синтаксис (->): Используется в switch expressions для компактности.
#Java #для_новичков #beginner #switch_case
👍3
3. Типы конструкций switch
3.1. Классический switch (оператор)
Используется для выполнения блока кода для одного или нескольких значений.
Требует break, чтобы остановить выполнение после совпадения.
Пример:
3.2. Switch с fall-through
Если не указать break, выполнение "проваливается" в следующий case.
Пример:
3.3. Switch Expression
Возвращает значение, которое можно присвоить переменной.
Использует -> для компактности или {} для сложной логики.
Пример:
3.4. Switch с enum
Удобен для работы с перечислениями (enum).
Пример:
4. Правильное применение switch
Чтобы писать понятный и эффективный код с switch, следуйте этим рекомендациям:
4.1. Используйте switch для четких вариантов
Применяйте switch, когда проверяете одно значение на несколько конкретных вариантов (например, дни недели, оценки).
Пример:
4.2. Не забывайте break в классическом switch
Без break код продолжит выполняться для всех последующих case, что может привести к ошибкам.
Пример ошибки:
4.3. Используйте switch expressions для простоты
Если нужно вернуть значение, используйте switch expression вместо классического switch.
Пример:
4.4. Добавляйте default
Всегда включайте default, чтобы обработать неожиданные значения.
Пример:
4.5. Проверяйте типы данных
Убедитесь, что выражение в switch соответствует поддерживаемому типу (int, String, enum и т.д.).
Пример ошибки:
#Java #для_новичков #beginner #switch_case
3.1. Классический switch (оператор)
Используется для выполнения блока кода для одного или нескольких значений.
Требует break, чтобы остановить выполнение после совпадения.
Пример:
char grade = 'B';
switch (grade) {
case 'A':
System.out.println("Отлично!");
break;
case 'B':
System.out.println("Хорошо!");
break;
case 'C':
System.out.println("Нормально");
break;
default:
System.out.println("Неизвестная оценка");
}
Вывод: Хорошо!
3.2. Switch с fall-through
Если не указать break, выполнение "проваливается" в следующий case.
Пример:
int month = 4;
switch (month) {
case 3:
case 4:
case 5:
System.out.println("Весна");
break;
default:
System.out.println("Другой сезон");
}
Вывод: Весна
3.3. Switch Expression
Возвращает значение, которое можно присвоить переменной.
Использует -> для компактности или {} для сложной логики.
Пример:
int number = 2;
int result = switch (number) {
case 1 -> 10;
case 2 -> {
int temp = number * 20;
yield temp; // Возвращает значение
}
default -> 0;
};
System.out.println(result);
Вывод: 40
3.4. Switch с enum
Удобен для работы с перечислениями (enum).
Пример:
enum Day { MONDAY, TUESDAY, WEDNESDAY }
Day day = Day.TUESDAY;
String message = switch (day) {
case MONDAY -> "Начало недели";
case TUESDAY -> "Второй день";
case WEDNESDAY -> "Середина недели";
};
System.out.println(message);
Вывод: Второй день
4. Правильное применение switch
Чтобы писать понятный и эффективный код с switch, следуйте этим рекомендациям:
4.1. Используйте switch для четких вариантов
Применяйте switch, когда проверяете одно значение на несколько конкретных вариантов (например, дни недели, оценки).
Пример:
int score = 85;
switch (score / 10) {
case 9:
System.out.println("Оценка: A");
break;
case 8:
System.out.println("Оценка: B");
break;
default:
System.out.println("Оценка ниже");
}
4.2. Не забывайте break в классическом switch
Без break код продолжит выполняться для всех последующих case, что может привести к ошибкам.
Пример ошибки:
int day = 1;
switch (day) {
case 1:
System.out.println("Понедельник");
case 2:
System.out.println("Вторник");
}
Вывод: Понедельник и Вторник (из-за отсутствия break).
4.3. Используйте switch expressions для простоты
Если нужно вернуть значение, используйте switch expression вместо классического switch.
Пример:
String fruit = "apple";
String type = switch (fruit) {
case "apple", "pear" -> "Фрукт";
case "carrot" -> "Овощ";
default -> "Неизвестно";
};
4.4. Добавляйте default
Всегда включайте default, чтобы обработать неожиданные значения.
Пример:
char letter = 'Z';
switch (letter) {
case 'A':
System.out.println("Первая буква");
break;
default:
System.out.println("Другая буква");
}
4.5. Проверяйте типы данных
Убедитесь, что выражение в switch соответствует поддерживаемому типу (int, String, enum и т.д.).
Пример ошибки:
double value = 1.5;
switch (value) { // Ошибка: double не поддерживается
case 1.5:
System.out.println("1.5");
}
#Java #для_новичков #beginner #switch_case
👍3
5. Работа switch под капотом
5.1. Компиляция в байт-код
Классический switch компилируется в одну из двух инструкций байт-кода:
tableswitch: Для последовательных значений case (например, 1, 2, 3).
lookupswitch: Для непоследовательных значений (например, 1, 10, 100).
Эти инструкции создают таблицу переходов, которая быстрее, чем многократные сравнения в if-else.
Пример:
Байт-код (упрощенно):
5.2. Switch Expressions
Switch expressions компилируются в байт-код с использованием invokedynamic (с Java 12+), что позволяет JVM оптимизировать выполнение.
Ключевое слово yield в switch expressions генерирует инструкцию для возврата значения.
5.3. Память и стек
Стек операндов: Значение в switch загружается в стек операндов JVM для сравнения с case.
Куча: Если switch работает с объектами (например, String или enum), их ссылки хранятся в стеке, а сами объекты — в куче.
Переменные: Локальные переменные внутри case хранятся в стеке вызовов.
5.4. Оптимизация в JVM
Таблицы переходов: tableswitch и lookupswitch быстрее, чем многократные сравнения в if-else, особенно для большого числа case.
JIT-компиляция: JIT-компилятор может встраивать switch в машинный код, улучшая производительность.
Оптимизация констант: Если switch использует константы, JVM может заранее вычислить переходы.
5.5. Ошибки в памяти
Проваливание (fall-through): Без break в классическом switch выполняется лишний код, что может привести к ошибкам.
Неправильные типы: Использование неподдерживаемых типов (например, double) вызывает ошибку компиляции.
Слишком много case: Большое количество case увеличивает размер байт-кода, но это редко влияет на производительность.
Пример ошибки:
6. Новшества в switch
Java 17 (сентябрь 2021, LTS)
Pattern Matching для switch (превью): Теперь switch может проверять не только значения, но и типы объектов. Это делает код короче и понятнее, чем if-else с instanceof.
Guarded Patterns (с &&): Можно добавлять дополнительные условия в case с помощью &&.
Обработка null: switch больше не выбрасывает NullPointerException для null, если есть case null.
Java 21 (сентябрь 2023, LTS)
Pattern Matching стал постоянной фичей: Улучшено и стабилизировано из Java 17. Теперь поддерживает сложные проверки типов и декомпозицию объектов (например, с record).
Guarded Patterns с when: Вместо && используется when для условий в case, что делает код читаемее.
Полная проверка покрытия: Для sealed классов или enum компилятор проверяет, что все случаи учтены, без default.
Java 25 (сентябрь 2025, Early Access)
Поддержка примитивных типов в Pattern Matching (превью): Теперь switch может работать с примитивными типами (int, long, double) в case с паттернами.
Безымянные переменные (_): Можно использовать _ для неиспользуемых переменных в паттернах, что упрощает код.
#Java #для_новичков #beginner #switch_case
5.1. Компиляция в байт-код
Классический switch компилируется в одну из двух инструкций байт-кода:
tableswitch: Для последовательных значений case (например, 1, 2, 3).
lookupswitch: Для непоследовательных значений (например, 1, 10, 100).
Эти инструкции создают таблицу переходов, которая быстрее, чем многократные сравнения в if-else.
Пример:
int day = 2;
switch (day) {
case 1:
System.out.println("Понедельник");
break;
case 2:
System.out.println("Вторник");
break;
}
Байт-код (упрощенно):
iload day
tableswitch {
1: invokevirtual // Вызов для "Понедельник"
2: invokevirtual // Вызов для "Вторник"
default: return
}
5.2. Switch Expressions
Switch expressions компилируются в байт-код с использованием invokedynamic (с Java 12+), что позволяет JVM оптимизировать выполнение.
Ключевое слово yield в switch expressions генерирует инструкцию для возврата значения.
5.3. Память и стек
Стек операндов: Значение в switch загружается в стек операндов JVM для сравнения с case.
Куча: Если switch работает с объектами (например, String или enum), их ссылки хранятся в стеке, а сами объекты — в куче.
Переменные: Локальные переменные внутри case хранятся в стеке вызовов.
5.4. Оптимизация в JVM
Таблицы переходов: tableswitch и lookupswitch быстрее, чем многократные сравнения в if-else, особенно для большого числа case.
JIT-компиляция: JIT-компилятор может встраивать switch в машинный код, улучшая производительность.
Оптимизация констант: Если switch использует константы, JVM может заранее вычислить переходы.
5.5. Ошибки в памяти
Проваливание (fall-through): Без break в классическом switch выполняется лишний код, что может привести к ошибкам.
Неправильные типы: Использование неподдерживаемых типов (например, double) вызывает ошибку компиляции.
Слишком много case: Большое количество case увеличивает размер байт-кода, но это редко влияет на производительность.
Пример ошибки:
String fruit = "apple";
switch (fruit) {
case "apple":
System.out.println("Фрукт");
case "banana": // Без break
System.out.println("Тоже фрукт");
}
Вывод: Фрукт и Тоже фрукт (из-за fall-through).
6. Новшества в switch
Java 17 (сентябрь 2021, LTS)
Pattern Matching для switch (превью): Теперь switch может проверять не только значения, но и типы объектов. Это делает код короче и понятнее, чем if-else с instanceof.
javaObject obj = "Hello";
String result = switch (obj) {
case Integer i -> "Это число";
case String s -> "Это строка";
default -> "Неизвестно";
};
Guarded Patterns (с &&): Можно добавлять дополнительные условия в case с помощью &&.
javacase Employee e && e.getDept().equals("IT") -> "IT сотрудник";
Обработка null: switch больше не выбрасывает NullPointerException для null, если есть case null.
Java 21 (сентябрь 2023, LTS)
Pattern Matching стал постоянной фичей: Улучшено и стабилизировано из Java 17. Теперь поддерживает сложные проверки типов и декомпозицию объектов (например, с record).
javarecord Point(int x, int y) {}
Object obj = new Point(1, 2);
String result = switch (obj) {
case Point(var x, var y) -> "Точка: " + x + ", " + y;
default -> "Не точка";
};
Guarded Patterns с when: Вместо && используется when для условий в case, что делает код читаемее.
javacase Employee e when e.getDept().equals("IT") -> "IT сотрудник";
Полная проверка покрытия: Для sealed классов или enum компилятор проверяет, что все случаи учтены, без default.
Java 25 (сентябрь 2025, Early Access)
Поддержка примитивных типов в Pattern Matching (превью): Теперь switch может работать с примитивными типами (int, long, double) в case с паттернами.
javaint value = 42;
String result = switch (value) {
case int i when i > 0 -> "Положительное число";
case int i -> "Ноль или отрицательное";
};
Безымянные переменные (_): Можно использовать _ для неиспользуемых переменных в паттернах, что упрощает код.
javacase Point(int x, _) -> "Только x: " + x;
#Java #для_новичков #beginner #switch_case
👍3
Как Вы считаете, оплата за обучение в кредит или частью будущей зарплаты - правильно?
Anonymous Poll
28%
Да, вполне, у каждого свой путь к мечте 👏
10%
Да, но выбора нет 😔
17%
Безразлично, я этим не балуюсь 🏝
31%
Нет, это не правильно 👎
14%
Я категорически против, они сволочи паразитирующие на людях! 😈
👍1
Что выведет код?
#Tasks
public class Task07082025 {
public static void main(String[] args) {
Object obj = "hello";
String result = switch(obj) {
case Integer i -> "int: " + i;
case String s when s.length() > 5 -> "long string";
case String s -> {
String temp = s.toUpperCase();
yield "str: " + temp + " (" + s + ")";
}
case null, default -> "other";
};
System.out.println(result);
}
}
#Tasks
👍2🤯1
Варианты ответа:
Anonymous Quiz
16%
Ошибка компиляции
8%
"long string"
68%
"str: HELLO (hello)"
8%
"other"
👍1
Что такое default конструктор? 🤓
Ответ:
Если в классе не определен конструктор, Java создает default конструктор без параметров, который инициализирует поля значениями по умолчанию (null, 0, false).
Пример:
class Example {
int x; // По умолчанию 0
}
Example e = new Example(); // Вызывает default конструктор
Если определен любой конструктор, default не создается.
#собеседование
Ответ:
Пример:
class Example {
int x; // По умолчанию 0
}
Example e = new Example(); // Вызывает default конструктор
Если определен любой конструктор, default не создается.
#собеседование
Please open Telegram to view this post
VIEW IN TELEGRAM
👍3
Влади́мир Бори́сович Бете́лин (род. 8 августа 1946, Кинешма Ивановской области) — советский и российский учёный, академик РАН, специалист в области информационных технологий; разработал инструментальные системы программного обеспечения и прикладные графические системы (САПР).
Михаил Владимирович Донской (8 августа 1948 — 13 января 2009, Москва) — российский программист, один из авторов первой советской шахматной программы «Каисса»
1973 — Первый микропроцессор в космосе (Pioneer Venus)
Начало использования микропроцессоров в космической технике.
1974 — DIY-компьютер Mark‑8
Публикация схемы одного из первых домашних ПК в Radio-Electronics. Этап на пути к персональным вычислениям.
1989 — Анонс Tandy PC
Компания Tandy Corporation начала продажи ПК под своим брендом, способствуя доступности компьютеров для массового пользователя.
1995 — IPO Netscape Communications
Одно из ключевых событий эры доткомов: популяризация браузеров и Интернета.
1999 — Представлен процессор AMD Athlon
Процессор стал прорывом в конкурентной гонке с Intel на рынке производительных CPU.
2004 — 1 миллион треков в iTunes Store
Apple достигла рубежа в 1 млн песен. Веха в развитии цифровой музыки и легальных медиаплатформ.
#Biography #Birth_Date #Events #8Августа
Please open Telegram to view this post
VIEW IN TELEGRAM
👍2