Модификаторы доступа

Видимость свойств влияет не только на внешнее поведение объектов, но и на отношения между наследуемыми классами. Публичные свойства и методы доступны всем наследникам. К ним можно обращаться как внутри объекта, так и снаружи:

<?php

class HTMLElement
{
    public $visible = true;

    public function isVisible()
    {
        return $this->visible;
    }
}

class DivElement extends HTMLElement
{
    public function isVisiblePropertyFromParent()
    {
        return $this->visible;
    }
    public function isVisibleMethodFromParent()
    {
        return $this->isVisible();
    }
}

$div = new DivElement();

// Вызов родительского свойства напрямую
echo $div->visible; // => true
// Вызов родительского метода напрямую
echo $div->isVisible(); // => true

// Вызов родительского свойства изнутри объекта
echo $div->isVisiblePropertyFromParent(); // => true

// Вызов родительского метода изнутри объекта
echo $div->isVisibleMethodFromParent(); // => true

Количество классов в цепочке наследования никак не влияет на это поведение. Любой подкласс DivElement, точно так же получит доступ к публичным частям HTMLElement:

<?php

class DivElementWithEmptyBody extends DivElement
{

}

$div = new DivElementWithEmptyBody();

// Вызов родительского свойства напрямую
echo $div->visible; // => true
// Вызов родительского метода напрямую
echo $div->isVisible(); // => true

Наследование не влияет на поведение свойств внутри объектов. Значение visible в каждом конкретном объекте связано только с этим объектом:

<?php

$div1 = new DivElementWithEmptyBody();
$div2 = new DivElement();

echo $div1->visible; // => true
echo $div2->visible; // => true

$div1->visible = false; // true
echo $div1->visible; // => false
echo $div2->visible; // => true

Приватные свойства и методы

Приватные свойства и методы, доступны только внутри того класса, где они были определены. Наследники не могут получить к ним доступ. Подразумевается, что приватные сущности это нечто персональное для класса, его внутренняя реализация, которую нельзя выставлять наружу. Однако, это не отменяет возможности взаимодействовать с приватными данными через публичный интерфейс.

<?php

class HTMLElement
{
    private $visible = true;

    public function isVisible()
    {
        return $this->visible;
    }
}

$div = new DivElement();
$div->isVisible(); // true
$div->visible; // Error

Защищённые свойства и методы

Они имеют самое необычное поведение, некая смесь между публичными и приватными. Ключевое слово protected используется тогда, когда разработчик хочет запретить доступ снаружи объекта, но дать возможность работать с ними внутри объекта класса-наследника или суперкласса. Звучит довольно туманно, посмотрим на практике:

<?php

class HTMLElement
{
    protected $visible = true;

    public function isVisible()
    {
        return $this->visible;
    }
}

class DivElement extends HTMLElement
{
    public function isVisiblePropertyFromParent()
    {
        return $this->visible;
    }
}

$div = new DivElement();

// Доступно внутри через родительский метод
$div->isVisible(); // true
// Доступно внутри напрямую
$div->isVisiblePropertyFromParent(); // true

// Недоступно снаружи
$div->visible; // Error

Выбор

Мы только начали знакомиться с наследованием, но уже сейчас видно, что всё не просто. Одну и ту же задачу можно сделать множеством способов. Какой предпочесть?

Универсальная стратегия, которой стоит придерживаться в большинстве случаев – всегда работать через абстракцию, пока она не мешает. Это значит что все свойства делаются приватными, а наружу выставляется публичный интерфейс (методы). Если из задачи очевидно, что публичный интерфейс нужен только наследникам и не должен использоваться снаружи объекта, то для этих методов ставится модификатор protected.

Мы учим программированию с нуля до стажировки и работы. Попробуйте наш бесплатный курс «Введение в программирование» или полные программы обучения по Javascript, PHP, Python и Java.

Хекслет

Подробнее о том, почему наше обучение работает →