В чём разница между JDK, JRE и JVM?
JVM — исполняет байткод; JRE = JVM + стандартные библиотеки (для запуска программ); JDK = JRE + компилятор javac и инструменты разработки (для разработки и сборки).
Разница между JDK, JRE и JVM
Эти три аббревиатуры описывают три разных уровня Java-платформы: от абстрактной спецификации исполнения до полного набора инструментов разработчика. Важно понимать, что начиная с Java 9 граница между JRE и JDK стала менее чёткой.
JVM — Java Virtual Machine
JVM — это спецификация виртуальной машины, которая:
- загружает и верифицирует байткод (
.class-файлы) - интерпретирует байткод или компилирует его в нативный код через JIT (Just-In-Time Compiler)
- управляет памятью через Garbage Collector
- обеспечивает безопасность и изоляцию кода
Популярные реализации JVM: HotSpot (Oracle/OpenJDK), GraalVM, Eclipse OpenJ9, Azul Zing. JVM не умеет запускать .java-файлы напрямую — только скомпилированный байткод.
JRE — Java Runtime Environment
JRE = JVM + стандартная библиотека классов (rt.jar в Java 8, или модули в Java 9+). Предназначен для запуска Java-программ, но не для их разработки. Содержит:
- JVM
- Стандартные библиотеки:
java.lang,java.util,java.io,java.netи т.д. - Конфигурационные файлы, шрифты, локали
Начиная с Java 11, Oracle прекратила отдельную публикацию JRE. Теперь для production-развёртывания создают минимальный runtime с помощью jlink.
JDK — Java Development Kit
JDK = JRE + инструменты разработки. Это то, что устанавливают разработчики. Содержит:
- JRE (или аналогичные компоненты)
javac— компилятор Java-исходников в байткодjar— упаковка в архивыjavadoc— генератор документацииjdb— отладчикjcmd,jstack,jmap,jstat— диагностические инструментыjlink— создание минимального кастомного runtimejshell— интерактивный REPL (с Java 9)
Визуальная схема вложения
JDK
+-- JRE
| +-- JVM (HotSpot, OpenJ9, GraalVM...)
| +-- Standard Libraries (java.*, javax.*)
+-- javac, jar, jlink, jshell, jcmd, jstack...
Практический пример: компиляция и запуск
# Компиляция (нужен JDK)
javac Hello.java # создаёт Hello.class
# Запуск (достаточно JRE/JVM)
java Hello # JVM загружает и выполняет байткод
# Создание минимального runtime для деплоя (JDK 9+)
jlink --add-modules java.base,java.net.http \
--output /opt/minimal-jre
/opt/minimal-jre/bin/java -jar app.jar
В контексте Docker
Для production-образов используют eclipse-temurin:21-jre (содержит только JRE) вместо eclipse-temurin:21-jdk, чтобы уменьшить размер образа. Для сборки в multi-stage build нужен JDK:
# Dockerfile
FROM eclipse-temurin:21-jdk AS build
COPY . .
RUN ./gradlew bootJar
FROM eclipse-temurin:21-jre AS runtime
COPY --from=build /app/build/libs/app.jar /app.jar
CMD ["java", "-jar", "/app.jar"]
Подводные камни
- С Java 11 Oracle не публикует отдельный JRE-дистрибутив — при необходимости JRE-образа используйте Temurin/Adoptium или создавайте через
jlink. - Разные дистрибутивы JDK (Oracle JDK, OpenJDK, GraalVM, Azul Zulu) имеют одинаковый API, но могут различаться лицензиями и производительностью GC.
- Версия JDK в CI и JRE в production должны совпадать — байткод Java 21 не запустится на JVM Java 17 (UnsupportedClassVersionError).
- JAVA_HOME нередко указывает на JRE вместо JDK на некоторых дистрибутивах Linux — это ломает Maven/Gradle, которым нужен
javac. - При использовании GraalVM Native Image JVM фактически отсутствует в runtime — приложение компилируется в нативный бинарник и не использует байткод.
- jshell входит в JDK, но не в JRE — частая ошибка при попытке запустить REPL в production-контейнере.
Common mistakes
- Путать термин «jdk jre jvm» с соседним механизмом Java.
- Не называть границу lifecycle, transaction, thread или request для «jdk jre jvm».
- Игнорировать production-эффекты «jdk jre jvm»: latency, SQL shape, memory, security или observability.
What the interviewer is testing
- Попросить объяснить механизм «jdk jre jvm» на минимальном примере.
- Проверить, видит ли кандидат failure mode и диагностику для «jdk jre jvm».
- Уточнить, какие настройки или API меняют «jdk jre jvm» в реальном сервисе.