Обратите внимание, что теперь используется ключевое слово implements вместо ключевого слова extends.
Для реализации интерфейса подразумевается определение класса, где для всех методов, дается реализация.
Мы можем также дополнительно определить поля и конструкторы, а также другие методы, возможно приватные.
Класс следует спецификации, заданной интерфейсом, и добавляет конкретные детали о том, как эти методы могут работать.
Однако нам не нужно идти от спецификации интерфейса к реализации класса за один шаг.
Мы могли бы также действовать поэтапно.
Путь от интерфейса к классу, который может быть создан, может быть короче или длиннее.
Во-первых, мы можем расширить один интерфейс от другого интерфейса, например, путем добавления абстрактных методов.
Мы можем частично реализовать интерфейс, путем реализации некоторых методов.
В этом случае мы получаем в результате не класс, а абстрактный класс, потому что не все методы реализованы.
И, наконец, мы можем перейти непосредственно от интерфейса к классу, путем реализации всех методов.
Если у нас есть абстрактный класс, мы можем расширить его другим абстрактным классом, например, если мы реализуем некоторые абстрактные методы, но не все из них.
Интерфейсы помогают нам моделировать системы, которые позволяют нам повторно использовать не только просто код, но и целиком концепции.
Теперь важно то, что класс может реализовать не только один, но и несколько интерфейсов.
В этом случае класс должен реализовать все методы от всех интерфейсов.
Помните, что класс не может расширять несколько классов.
В Java нет множественного наследования, как в других языках программирования, таких как C ++.
Класс не может расширять два класса.
Однако в Java класс может реализовать два интерфейса.
Это способ сказать, что A является B и C.
Например, мы можем определить амфибию как реализацию интерфейса автомобиля и интерфейса лодки.
Этот амфибия-автомобиль будет реализовывать методы обоих интерфейсов.
Теперь, при этом, могут возникнуть конфликты имен.
Что произойдет, если у нас есть одно и тоже имя метода в обоих интерфейсах?
Если у нас есть одинаковое имя метода в обоих интерфейсах, но разные возвращаемые типы, тогда возникает ошибка.
Если имя метода и тип возвращаемого значения совпадают, тогда все в порядке.
Если, кроме того, совпадают и параметры методов, тогда класс должен реализовать этот метод один раз.
Они неразличимы.
Если типы параметров методов не совпадают, мы имеем случай перегрузки, который обрабатывается таким же образом, как будто эти методы принадлежат одному интерфейсу.
Оба эти методы должны быть реализованы.
Мы можем также определить класс путем реализации интерфейса и наследования от класса одновременно.
Также нужно сказать, что помимо абстрактных классов и интерфейсов для структурирования кода используются классы-утилиты, в которых определены только статические члены.
Как правило, такие классы используются для объединения родственных алгоритмов.
Примером такого класса может служить класс Math, предоставляющий реализации различных математических функций.
Таким образом, отношения реализации и наследования помогают нам определить структуры связанных интерфейсов и классов, которые помогают нам масштабировать и упорядочить код.
Проектирование интерфейсов всегда было непростой задачей, потому что, если мы хотим добавить дополнительные методы в интерфейсы, это потребует изменений во всех классах реализации.
По мере старения интерфейса количество реализующих его классов может увеличиться настолько, что будет невозможно расширить интерфейс, изменяя все классы реализации.
Вот почему большинство библиотек сначала обеспечивают базовый класс реализации, а затем они расширяют его и переопределяют его методы, чтобы при изменении интерфейса, изменить только базовый класс реализации.
В Java 8 вводится понятие метода по умолчанию интерфейса.
Для создания метода по умолчанию в интерфейсе нам нужно использовать ключевое слово «default» в сигнатуре метода.
Читать дальше