Введение
В начале карьеры, я не понимал, зачем нужен объектно-ориентированный подход. У нас уже есть функции, они решают проблему структурирования кода! В чем может быть проблема? Давайте разбираться.
Проверки ошибок опускаем для краткости. В промышленном коде они обязательны!
Напишем программу в процедурном стиле. Как пример, возьмем публичный API restcountries.com, получим по нему страну и на основе ее данных рассчитаем плотность населения.
function getCountry($name)
{
$ch = curl_init();
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
$url = 'https://restcountries.com/v3.1/name/';
curl_setopt($ch, CURLOPT_URL, $url . $name);
$result=curl_exec($ch);
curl_close($ch);
return json_decode($result, true);
}
function getCountryDensity($countryName)
{
$country = getCountry($countryName);
$population = $country[0]['population'];
$area = $country[0]['area'];
$density = $population / $area;
if ($density < 100) {
return 'low';
} else if ($density < 300) {
return 'medium';
} else {
return 'high';
}
}
echo getCountryDensity('uk');
Границы для low, medium и high я взял из головы, на основе сравнения нескольких стран. Пусть это будет нашей доменной логикой.
Наш код работает, в нем грамотно распределены зоны ответственности: есть источник данных (getCountry), есть доменная логика (getCountryDensity). Такой подход подойдет для одноразовых скриптов или коротких программ, но с ним сложно работать в сколько-нибудь серьезных проектах.
Когда несколько человек работает над проектом одновременно, сложность кода возрастает стремительно. Паутину функций быстро становится сложно понять. Далее мы увидим, как можно разделить программу на небольшие изолированные модули: классы.
От ассоциативного массива к классу
Начнем с того, что мы не знаем, какие поля и типы данных может содержать ответ. Давайте опишем их, создадим класс Country 1. Такой класс называется “модель данных”. Элементы массива станут свойствами (property) класса.
Иногда можно встретить название CountryDTO. DTO означает Data Transfer Object. Это говорит нам о том, что объект получили из внешнего источника (или наоборот) и он не содержит логики.
class Country
{
/** @var string */
public $name;
/** @var float */
public $area;
/** @var int */
public $population;
}
function getCountry($name)
{
// ...
$countryArr = json_decode($result, true);
$country = new Country();
$country->name = $countryArr[0]['name']['common'];
$country->area = $countryArr[0]['area'];
$country->population = $countryArr[0]['population'];
return $country;
}
function getCountryDensity($countryName)
{
$country = getCountry($countryName);
$density = $country->population / $country->area;
// ...
}
Теперь нам не нужно каждый раз делать сетевой запрос, чтобы увидеть структуру объекта (по крайней мере, интересующей нас его части). Как альтернатива, можно было бы описать комментариями, какие поля вернет запрос, но для этого пришлось бы строить дополнительные ассоциации в уме, а это увеличивает когнитивную нагрузку.
От функции к классу
Так как источник данных вынесен в отдельную функцию, при необходимости, мы можем переехать на другой: вместо API читать данные из БД или файла. Но наш код негибок: для выполнения функции getCountryDensity
всегда нужен API. В он не всегда доступен.
Здесь нам и понадобится ООП. Для начала, выделим функции в отдельные самодостаточные модули - классы. Функции становятся методами классов (method).
class CountryRepository
{
public function findByName($name)
{
// ...
$countryArr = json_decode($result, true);
$country = new Country();
$country->name = $countryArr[0]['name']['common'];
$country->area = $countryArr[0]['area'];
$country->population = $countryArr[0]['population'];
return $country;
}
}
class CountryService
{
/** @var CountryRepository */
private $countries;
/** @param CountryRepository $countries */
public function __construct($countries)
{
$this->countries = $countries;
}
public function getCountryDensity($countryName)
{
$country = $this->countries->findByName($countryName);
// ...
}
}
$countries = new CountryRepository();
$countryService = new CountryService($countries);
echo $countryService->getCountryDensity('uk');
У ООП есть порог входа: программу стало сложнее понять, она занимает больше строк. Но взамен мы получаем модульность, основу качественного программирования. Каждый класс представляет собой небольшую законченную программу.
От класса к интерфейсу
Сейчас класс CountryService
все еще полагается на CountryRepository
и использует API. Что делать? Используем интерфейс (interface), то есть контракт, который класс должен выполнить. Если класс реализует (implements) интерфейс CountryRepositoryInterface
, то у него должен иметься метод findByName
с правильными параметрами и модификатором доступа. Каким образом он получает данные становится не важно:
interface CountryRepositoryInterface
{
public function findByName($name);
}
class ApiCountryRepository implements CountryRepositoryInterface
{
// ...
}
class DatabaseCountryRepository implements CountryRepositoryInterface
{
// ...
}
class FileCountryRepository implements CountryRepositoryInterface
{
// ...
}
class CountryService
{
/** @var CountryRepositoryInterface */
private $countries;
/** @param CountryRepositoryInterface $countries */
public function __construct($countries)
{
$this->countries = $countries;
}
// ...
}
Такой прием называется полиморфизмом. CountryService
больше не зависит от конкретной реализации CountryRepository
. Он будет работать с любым, кто реализует интерфейс CountryRepositoryInterface
. Где это можно использовать:
- при написании модульных тестов,
- при переезде на другую БД,
- когда приходит бизнес и говорит, что все надо переделать.
Интерфейсы дают делают систему гибкой.
Встречаются интерфейсы без суффикса Interface в названии. Мы его используем в соответствии с требованиями PSR Naming Conventions.
Давайте подменим CountryRepository
на тестовую реализацию, которая хранит страны в памяти.
class FakeCountryRepository implements CountryRepositoryInterface
{
/** @var Country[] */
private $countries;
public function __construct($countries)
{
$this->countries = $countries;
}
public function findByName($name)
{
foreach ($this->countries as $country) {
if ($country->name === $name) {
return $country;
}
}
return null;
}
}
$uk = new Country();
$uk->name = 'uk';
$uk->area = 10000.0;
$uk->population = 3000;
$countries = new FakeCountryRepository([$uk]);
$countryService = new CountryService($countries);
echo $countryService->getCountryDensity('uk');
Больше нам не нужно иметь полноценный API для проверки нашего кода. Проверять работу API все еще нужно, но это уже другая история.
Назначение обязанностей
В нашем коде еще есть возможность для улучшения: расчет плотности требует только данные объекта Country
. Давайте назначим ему эту обязанность.
Заодно закроем доступ извне к свойствам объекта. В ООП мы даем доступ к ним только через методы. Такой прием называется инкапсуляцией. Он нужен для защиты от несанкционированных изменений: становится проще понять, что может содержать свойство.
class Country
{
private $name;
private $area;
private $population;
// ...
public function getName()
{
return $this->name;
}
public function getDensity()
{
$density = $this->population / $this->area;
if ($density < 100) {
return 'low';
} else if ($density < 300) {
return 'medium';
} else {
return 'high';
}
}
}
class CountryService
{
// ...
public function getCountryDensity($countryName)
{
$country = $this->countries->findByName($countryName);
return $country->getDensity();
}
}
Подобный подход называется Information Expert: “Обязанности должны быть назначены объекту, который владеет максимумом необходимой информации для выполнения обязанности”.
Наш код стал модульным и гибким:
- Country не зависит ни от чего, знает как рассчитывать значения на основе своих данных;
- CountryRepository отвечает только за получение стран по API, мы без проблем можем заменить его;
- CountryService служит как посредник и содержит очень мало логики.
Заключение
Мы не трогали поведение кода, при этом изменили его структуру. Такой процесс называется рефакторингом. В результате мы получили более модульный код. Отдельные классы не зависят друг от друга. В любое время мы можем изменить способ получения данных, при этом не внося изменения в код клиента.
Реальные процедурные программы бывает значительно сложнее понять, они сильно связаны между собой. ООП позволяет нам ограничить сложность, провести “швы” между модулями программы.
ООП сложно понять на первый взгляд: программа как будто состоит из делегирования методов. Но в разделении ответственности и заключается его мощь.
Часто ли приходится менять источники данных? Нет. Иногда происходит переезд с одной базы данных на другую, но это - редкий случай. Гораздо чаще этот прием используется в модульном тестировании. Но про тестирование в другой раз.
Не торопимся кидать тапки за публичные свойства, мы к этому вернемся. ↩︎