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

Работа с несколькими файлами Java: Настройка окружения

Реальные программы на Java состоят из сотен и тысяч файлов. Их уже не запустишь одной командой java. Процесс работы в таком случае выглядит сложнее. Предположим, что у нас есть два файла One.java и Two.java с таким содержимым:

// io/hexlet/One.java
// Этот файл точка входа
package io.hexlet;

public class One {
    public static void main(String[] args) {
        Two.greeting();
    }
}

// io/hexlet/Two.java
package io.hexlet;

public class Two {
    public static void greeting() {
        System.out.println("Java for Brave");
    }
}

В первом файле реализован метод main() так как это точка входа, во втором – обычный статический метод, который печатает на экран текст. Запуск java One.java приведет к ошибке:

One.java:5: error: cannot find symbol
        Two.greeting();
        ^
  symbol:   variable Two
  location: class One
1 error
error: compilation failed

То есть компиляция завершается с ошибкой. Что произойдет если мы попытаемся скомпилировать первый файл самостоятельно?

javac One.java

One.java:5: error: cannot find symbol
        Two.greeting();
        ^
  symbol:   variable Two # на чем произошла ошибка
  location: class One # место где произошла ошибка
1 error

Возникнет ровно та же самая ошибка. java внутри себя запускает javac. Компилятор проверяет код на типобезопасность, а для этого ему нужна информация обо всех используемых классах. Класс Two компилятору не известен, поэтому процесс останавливается. Дальше можно поступить двумя способами:

  1. Компилировать по очереди все файлы, начиная с тех, которые не зависят ни от кого, постепенно поднимаясь до тех класов, в которых используются все остальные
  2. Сразу передать компилятору весь список файлов
// Все это в директории io/hexlet
// Сработает только такой порядок
javac Two.java
javac -classpath ../.. One.java
// Про параметр classpath будет рассказано ниже

// Или проще так. В любом порядке, компилятор разберется сам
javac One.java Two.java

Теперь попробуем запустить:

# Внутри io/hexlet
java One

Error: Could not find or load main class One
Caused by: java.lang.ClassNotFoundException: One

А вот это уже неожиданно. Почему класс One не найден? Когда класс определяется внутри пакета, то при обращении к нему нужно указывать полное имя. В нашем случае io.hexlet.One. Но даже если мы укажем имя полностью, то все равно получим ошибку:

# Внутри io/hexlet
java io.hexlet.One

Error: Could not find or load main class io.hexlet.One
Caused by: java.lang.ClassNotFoundException: io.hexlet.One

Как и положено, java пытается найти класс по пути io/hexlet/One.class, но мы уже находимся внутри io/hexlet, поэтому возникает ошибка. Если выйти в родительскую директорию io, то программа заработает:

# Внутри родительской директории io
java io.hexlet.One
Java for Brave

Вот теперь работает! Для этого пришлось выйти из директории, дав возможность java найти класс по полному имени. Но можно и не выходить. Для этого нам понадобится classpath.

classpath

classpath – параметр, который задает директории для поиска классов. В нашем случае это позволит не выходить из директории:

# io/hexlet
# ../.. указывает на родительскую директорию io
java -classpath ../.. io.hexlet.One
# или так java -cp ../.. io.hexlet.One
# -cp это сокращенная версия опции -classpath
Java for Brave

classpath имеет очень важное значение в Java. Классов много, все они находятся в разных пакетах, а значит и директориях. Поэтому при компиляции проектов невозможно обойтись без указания путей поиска классов. А неправильная работа с classpath, одна из самых распространенных ошибок новичков.

Перекомпиляция

Когда файлов мало, то их можно компилировать сразу все на любое изменение. Но с определенного размера, компиляция начнет занимать серьезное время. Тогда возникает вопрос, можно ли компилировать не все файлы, а только те, которые изменились? Да можно, но с ограничениями. Если изменилось что-то, что используется другим классом (а значит файлом), например имя класса, количество параметров метода и тому подобное, то придется по цепочке компилировать все связанные классы. Такая техника называется инкрементальная компиляция и современные инструменты умеют делать ее сами. Скоро об этом поговорим.


Самостоятельная работа

  1. В проекте hexlet-java создайте директорию io/hexlet
  2. В директории io/hexlet создайте файл One.java и добавьте туда код:

    package io.hexlet;
    
    public class One {
        public static void main(String[] args) {
            Two.greeting();
        }
    }
    
  3. В директории io/hexlet создайте файл Two.java и добавьте туда код:

    package io.hexlet;
    
    public class Two {
        public static void greeting() {
            System.out.println("Java for Brave");
        }
    }
    
  4. Скомпилируйте файлы и запустите программу на выполнение, находясь в директории io/hexlet. Убедитесь, что на экран вывелась строчка "Java for Brave".

  5. Добавьте файлы с расширением .class в .gitignore и залейте все изменения на гитхаб.


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

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

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

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

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

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

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

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

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

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

Логотип компании Альфа Банк
Логотип компании Aviasales
Логотип компании Yandex
Логотип компании Tinkoff
Рекомендуемые программы
профессия
от 25 000 ₸ в месяц
Разработка приложений на языке Java
10 месяцев
с нуля
Старт 23 января

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

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

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

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