среда, 10 февраля 2016 г.

Что такое ООП? Принципы ООП.

      Итак, для разминки начнем пожалуй с самого банального вопроса на собеседовании на позицию Java developer. Ничего нового я придумывать не буду, просто копипаст с понравившейся статьи. 

Немного теории:

Объектно-ориентированное программирование (в дальнейшем ООП) — парадигма программирования, в которой основными концепциями являются понятия объектов и классов.
В центре ООП находится понятие объекта.
Объект — это сущность, экземпляр класса, которой можно посылать сообщения, и которая может на них реагировать, используя свои данные. Данные объекта скрыты от остальной программы. Сокрытие данных называется инкапсуляцией.
Наличие инкапсуляции достаточно для объектности языка программирования, но ещё не означает его объектной ориентированности — для этого требуется наличие наследования.
Но даже наличие инкапсуляции и наследования не делает язык программирования в полной мере объектным с точки зрения ООП. Основные преимущества ООП проявляются только в том случае, когда в языке программирования реализован полиморфизм; то есть возможность объектов с одинаковой спецификацией иметь различную реализацию.
Хочу выделить что очень часто натыкаюсь на мнение, что в ООП стоит выделять еще одну немаловажную характеристику — абстракцию. Официально ее не вносили в обязательные черты ООП, но списывать ее со счетов не стоит.
Абстрагирование — это способ выделить набор значимых характеристик объекта, исключая из рассмотрения не значимые  Соответственно, абстракция — это набор всех таких характеристик.
Инкапсуляция — это свойство системы, позволяющее объединить данные и методы, работающие с ними в классе, и скрыть детали реализации от пользователя.
Наследование — это свойство системы, позволяющее описать новый класс на основе уже существующего с частично или полностью заимствующейся функциональностью. Класс, от которого производится наследование, называется базовым, родительским или суперклассом. Новый класс — потомком, наследником или производным классом
Полиморфизм — это свойство системы использовать объекты с одинаковым интерфейсом без информации о типе и внутренней структуре объекта.

Инкапсуляция.

Инкапсуляция позволит скрыть детали реализации, и открыть только то что необходимо в последующем использовании. Другими словами инкапсуляция - это механизм контроля доступа.
images
Зачем же это нужно?
Думаю вам бы не хотелось что бы кто-то, что-то изменял в написанной вами библиотеки.
И если это опытный программист то это простить еще можно, но все равно не приятно, а вот если это начинающий или не осторожный который с легкой руки задумает изменить код, да еще не в ту степь, нам ведь такого не хочется !) Что бы обезопасить себя от таких поступков существует инкапсуляция.
Цель инкапсуляции — уйти от зависимости внешнего интерфейса класса(то , что могут использовать другие классы) от реализации. Чтобы малейшее изменение в классе не влекло за собой изменение внешнего поведения класса. Давайте разcмотрим как ею пользоваться.
Существует 4 вида модификаторов доступа: publicprotectedprivate и default.
Public - уровень  предполагает  доступ к компоненту с этим модификатором из экземпляра любого класса и любого пакета.
Protected - уровень  предполагает  доступ к компоненту с этим модификатором из экземпляров родного класса и классов-потомков, независимо от того в каком пакете они находятся.
Default — уровень предполагает  доступ к компоненту с этим модификатором из экземпляров любых классов , находящихся в одном пакете с этим классом.
Private - уровень предполагает  доступ к компоненту с этим модификатором только из этого класса.
?
1
2
3
4
5
6
public class Human {
    public String name;
    protected String surname;
    private int age;
    int birthdayYear;
}
public String name; — имя которое доступное из любого места в приложении.
protected String surname; — фамилия доступна из родного класса и потомков.
private int age; — возраст доступен только в рамках класса Human.
int birthdayYear; — хоть не указывается явный модификатор доступа, система понимает его как default, год рождения будет доступен всему пакету в котором находится класс Human.
Для разных структурных элементов класса предусмотрена возможность применять только определенные уровни модификаторов доступа.
Для класса- только public и default.
Для атрибутов класса — все 4 вида.
Для конструкторов — все 4 вида.
Для методов — все 4 вида.

Наследование.

Наследование — это процесс, посредством которого один объект может приобретать свойства другого. Точнее, объект может наследовать основные свойства другого объекта и добавлять к ним черты, характерные только для него.
Наследование является важным, поскольку оно позволяет поддерживать концепцию иерархии классов (hierarchical classification). Применение иерархии классов делает управляемыми большие потоки информации.
Разберем этот механизм на классическом примере: Геометрические фигуры.
1
У нас есть интерфейс Figure:
?
1
2
3
4
5
6
7
public interface Figure {
    public void draw ();
    public void erase ();
    public void move ();
    public String getColor ();
    public boolean setColor ();
}
Интерфейс (более детально будут рассмотрены в скором будущем) — нам говорит как должен выглядеть класс, какие методы в себе содержать, какими переменными и типами данных манипулировать. Сам интерфейс не реализует методы, а создает как бы скелет для класса который будет расширять этот интерфейс. Есть класс Figure который расширяет интерфейс Figure:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class Figure implements devcolibri.com.oop.inheritance.interfaces.Figure{
    @Override
    public void draw() {
       //need to implement
    }
    @Override
    public void erase() {
       //need to implement
    }    
    @Override      
    public void move(int pixel) {        
       //need to implement   
    }    
    @Override    
    public String getColor() {        
        return null;
    }    
    @Override    
    public boolean setColor(String colour) {        
        return false;
    }
}
В этом классе мы реализовываем все методы интерфейса Figure.
public class Figure implements devcolibri.com.oop.inheritance.interfaces.Figure — с помощью ключевого слова implements мы перенимаем методы интерфейса Figure для реализации.
Важно: в классе должны быть все методы интерфейса, даже если некоторые еще не реализованы, в противном случае компилятор будет выдавать ошибку и просить подключить все методы. Тело методов можно изменить только в интерфейсе, здесь только реализация.
@Override — аннотация которая говорит что метод переопределен.
И соответственно у нас есть 3 класса саммых фигур которые наследуются от класса Figure. Класс Figure является родительским классом или классом родителем, а классы Circle, Rectungle и  Triangle — являются дочерними.
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class Circle extends Figure {
    @Override
    public void draw() {
       super.draw();
    }
   @Override
   public void erase() {
      super.erase();
   }
   @Override
   public void move(int pixel) {
       super.move(pixel);
   }
   @Override
   public String getColor() {
       return super.getColor();
   }
   @Override
   public boolean setColor(String colour) {
       return super.setColor(colour);
   }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
public class Rectangle extends devcolibri.com.oop.inheritance.Figure{
    @Override
    public void draw() {
      super.draw();
    }
    @Override
    public void erase() {
       super.erase();
    }
    @Override
    public void move(int pixel) {
       super.move(pixel);
    }
    @Override
    public String getColor() {
       return super.getColor();
    }
    @Override
    public boolean setColor(String colour) {
       return super.setColor(colour);
   }
}
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
public class Triangle extends devcolibri.com.oop.inheritance.Figure{
    @Override
    public void draw() {
       super.draw();
    }
    @Override
    public void erase() {
       super.erase();
   
    @Override
    public void move(int pixel) {
       super.move(pixel);
    }
    @Override
    public String getColor() {
       return super.getColor();
    }
    @Override
    public boolean setColor(String colour) {
        return super.setColor(colour);
    }
}
public class Triangle extends devcolibri.com.oop.inheritance.Figure — это значит что класс Triangle наследует класс Figure.
super.setColor(colour); — super модификатор позволяющий вызывать методы из класса родителя.
Теперь каждый класс перенял свойства класса Figure. Что собственно это нам дало?
Значительно уменьшило время разработки классов самих фигур, дало доступ к полям и методам родительского класса.
Наверное возник вопрос: чем же extends отличается от implements?
Extends дает нам намного гибче подход. Мы используем только те методы что нам нужны, в любой момент мы можем изменить каркас и тело метода, или добавить совсем новый метод который возможно будет использовать информацию от класса родителя, а implements все лишь формирует тело класса.
В дочерних классах мы можем спокойно добавлять новые интересующие нас методы. Например мы хотим добавить в класс Triangle 2-а новых метода: flimHorizontal () и flipVertical ():
2
?
1
2
3
4
5
6
7
8
9
10
11
12
13
/**
* New Method
*/
public void flipVertical () {
};
/**
* New Method
*/
public void flipHorizontal () {
};
Теперь эти 2-а метода принадлежат сугубо классу Triangle. Этот подход используется когда базовый класс не может решить всех проблем.
Или можно использовать другой подход, изменить или переписать методы в дочерним классе:
3
Довольно интересный факт: в java запрещено множественное наследование, но любой из классов по умолчанию наследуется от класса Object.

Полиморфизм.

В более общем смысле, концепцией полиморфизма является идея «один интерфейс, множество реализаций«.
Это означает, что можно создать общий интерфейс для группы близких по смыслу действий. Преимуществом полиморфизма является то, что он помогает снижать сложность программ, разрешая использование того же интерфейса для задания единого класса действий. Выбор же конкретного действия, в зависимости от ситуации, возлагается на компилятор.
Вам, как программисту, не нужно делать этот выбор самому. Нужно только помнить и использовать общий интерфейс.
?
1
2
3
4
5
6
7
public class Parent {
    int a = 2;
}
public class Child extends Parent {
    int a = 3;
}
Прежде всего, нужно сказать, что такое объявление корректно.
Наследники могут объявлять поля с любыми именами, даже совпадающими с родительскими. Объекты класса Child будут содержать сразу две переменных, а поскольку они могут отличаться не только значением, но и типом (ведь это два независимых поля), именно компилятор будет определять, какое из значений использовать.
Компилятор может опираться только на тип ссылки, с помощью которой происходит обращение к полю:
?
1
2
3
4
Child c = new Child();
System.out.println(c.a); // результатом будет 3
Parent p = c;
System.out.println(p.a); //результатом будет 2
Обе ссылки указывают на один и тот же объект, но тип у них разный. Отсюда и результат. Объявление поля в классе-наследнике «скрыло» родительское поле.
Данное объявление так и называется – «скрывающим». Родительское поле продолжает существовать.
К нему можно обратиться явно:
?
1
2
3
4
5
class Child extends Parent {
     int a = 3;                     //скрывающее объявление
     int b = ((Parent)this).a;      //громоздкое обращение к родительскому полю
     int c = super.a;               //простое обращение к родительскому полю
}
Переменные b и получат значения, родительского поля a. Хотя выражение с super более простое, оно не позволит обратиться на два уровня вверх по дереву наследования.
А ведь вполне возможно, что в родительском классе это поле также было скрывающим и в родителе родителя храниться еще одно значение.
К нему можно обратиться явным приведением, как это делается для b.
?
1
2
3
4
5
6
7
8
9
10
class Parent {
     int x = 0;
     public void printX() {
          System.out.println(x);
     }
}
class Child extends Parent {
     int x = -1;
}
Каков будет результат для new Child.printX(); ?
Метод вызывается с помощью ссылки типа Child, но метод определен в классеParent и компилятор расценивает обращение к полю в этом методе именно как к полю класса Parent. Результатом будет 0.

Рассмотрим случай переопределения методов:
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class Parent {
     public int getValue() {
          return 0;
     }
}
class Child extends Parent {
     public int getValue() {
          return 1;
     }
}
Child c = new Child();
System.out.println(c.getValue()); // результатом будет 1
Parent p = c;
System.out.println(p.getValue()); // результатом будет 1
Родительский метод полностью перекрыт.
В этом ключевая особенность полиморфизма – наследники могут изменить родительское поведение, даже если обращение к ним производиться по ссылке родительского типа.
Хотя старый метод снаружи недоступен, внутри класса-наследника к нему можно обратиться с помощью super.
Статические методы, подобно статическим полям принадлежат классу и появление наследников на них не сказывается. Статические методы не могут перекрывать обычные методы и наоборот.

Абстракция:

Как говорилось в начале статьи нельзя игнорировать абстракцию, а значит и абстрактные классы и методы.
В контексте ООП абстракция - это обобщение данных и поведения для типа, находящегося выше текущего класса по иерархии.
Перемещая переменные или методы из подкласса в супер класс, вы обобщаете их. Это общие понятия, и они применимы в языке Java. Но язык добавляет также понятия абстрактных классов и абстрактных методов.
Абстрактный класс является классом, для которого нельзя создать экземпляр.
Например, вы можете создать класс Animal(животное). Нет смысла создавать экземпляр этого класса: на практике вам нужно будет создавать экземпляры конкретных классов, например, Dog (собака). Но все классы Animal имеют некоторые общие вещи, например, способность издавать звуки. То, чтоAnimal может издавать звуки, еще ни о чем не говорит.
Издаваемый звук зависит от вида животного.
Как это смоделировать?
Определить общее поведение в абстрактном классе и заставить подклассы реализовывать конкретное поведение, зависящее от их типа.
В иерархии могут одновременно находиться как абстрактные, так и конкретные классы.
Использование абстракции:
Наш класс Person содержит некоторый метод поведения, и мы пока не знаем, что он нам необходим. Удалим его и заставим подклассы реализовывать это поведение полиморфным способом. Мы можем сделать это, определив методы Person как абстрактные. Тогда наши подклассы должны будут реализовывать эти методы.
?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
public abstract class Person {
    abstract void move();
    abstract void talk();
}
public class Adult extends Person {
    public Adult() {
    }
    public void move() {
        System.out.println("Walked.");
    }
    public void talk() {
        System.out.println("Spoke.");
    }
}
public class Baby extends Person {
    public Baby() {
    }
    public void move() {
        System.out.println("Crawled.");
    }
    public void talk() {
        System.out.println("Gurgled.");
    }
}
Что мы сделали в приведенном выше коде?
Мы изменили Person и указали методы как abstract, заставив подклассы реализовывать их.
Мы сделали Adult подклассом Person и реализовали эти методы.
Мы сделали Baby подклассом Person и реализовали эти методы.
Объявляя метод абстрактным, вы требуете от подклассов либо реализации этого метода, либо указания метода в этих подклассах абстрактным и передачи ответственности по реализации метода к следующим подклассам. Можно реализовать некоторые методы в абстрактном классе и заставить подклассы реализовывать остальные. Это зависит от вас. Просто объявите методы, которые не хотите реализовывать, как абстрактные и не предоставляйте тело метода. Если подкласс не реализует абстрактный метод супер класса, компилятор выдаст ошибку.
Теперь, поскольку Adult и Baby являются подклассами Person, мы можем обратиться к экземпляру каждого класса как к типу Person.

Ссылка на оригинал

Довольно неплохой

Комментариев нет:

Отправить комментарий