В чем разница между публичным, частным и защищенным?

Когда и почему я должен использовать public, private, а также protected функции и переменные внутри класса? В чем разница между ними?

Примеры:

// Public
public $variable;
public function doSomething() {
  // ...
}

// Private
private $variable;
private function doSomething() {
  // ...
}

// Protected
protected $variable;
protected function doSomething() {
  // ...
}

16 ответов

Решение

Ты используешь:

  • public область действия, чтобы сделать эту переменную / функцию доступной из любого места, других классов и экземпляров объекта.

  • private область видимости, когда вы хотите, чтобы ваша переменная / функция была видна только в своем собственном классе.

  • protected область видимости, когда вы хотите сделать вашу переменную / функцию видимой во всех классах, которые расширяют текущий класс, включая родительский класс.

Подробнее: (для полной информации)

dd

Общественность:

Когда вы объявляете метод (функцию) или свойство (переменную) как public эти методы и свойства могут быть доступны:

  • Тот же класс, который объявил это.
  • Классы, которые наследуют объявленный выше класс.
  • Любые посторонние элементы вне этого класса также могут получить доступ к этим вещам.

Пример:

<?php

class GrandPa
{
    public $name='Mark Henry';  // A public variable
}

class Daddy extends GrandPa // Inherited class
{
    function displayGrandPaName()
    {
        return $this->name; // The public variable will be available to the inherited class
    }

}

// Inherited class Daddy wants to know Grandpas Name
$daddy = new Daddy;
echo $daddy->displayGrandPaName(); // Prints 'Mark Henry'

// Public variables can also be accessed outside of the class!
$outsiderWantstoKnowGrandpasName = new GrandPa;
echo $outsiderWantstoKnowGrandpasName->name; // Prints 'Mark Henry'

Защищено:

Когда вы объявляете метод (функцию) или свойство (переменную) как protected эти методы и свойства доступны

  • Тот же класс, который объявил это.
  • Классы, которые наследуют объявленный выше класс.

Члены посторонних не могут получить доступ к этим переменным. "Посторонние" в том смысле, что они не являются объектными экземплярами самого объявленного класса.

Пример:

<?php

class GrandPa
{
    protected $name = 'Mark Henry';
}

class Daddy extends GrandPa
{
    function displayGrandPaName()
    {
        return $this->name;
    }

}

$daddy = new Daddy;
echo $daddy->displayGrandPaName(); // Prints 'Mark Henry'

$outsiderWantstoKnowGrandpasName = new GrandPa;
echo $outsiderWantstoKnowGrandpasName->name; // Results in a Fatal Error

Точная ошибка будет такой:

Неустранимая ошибка PHP: невозможно получить доступ к защищенному свойству GrandPa:: $ name


Частный:

Когда вы объявляете метод (функцию) или свойство (переменную) как private эти методы и свойства могут быть доступны:

  • Тот же класс, который объявил это.

Члены посторонних не могут получить доступ к этим переменным. Аутсайдеры в том смысле, что они не являются экземплярами объекта самого объявленного класса и даже классов, которые наследуют объявленный класс.

Пример:

<?php

class GrandPa
{
    private $name = 'Mark Henry';
}

class Daddy extends GrandPa
{
    function displayGrandPaName()
    {
        return $this->name;
    }

}

$daddy = new Daddy;
echo $daddy->displayGrandPaName(); // Results in a Notice 

$outsiderWantstoKnowGrandpasName = new GrandPa;
echo $outsiderWantstoKnowGrandpasName->name; // Results in a Fatal Error

Точные сообщения об ошибках будут:

Примечание: неопределенное свойство: папа:: $ name
Неустранимая ошибка: невозможно получить доступ к частной собственности GrandPa:: $ name


Рассекая Дедушку с помощью отражения

Эта тема на самом деле не выходит за рамки, и я добавляю ее сюда, чтобы доказать, что рефлексия действительно мощная. Как я уже говорил в приведенных выше трех примерах, protected а также private члены (свойства и методы) не могут быть доступны вне класса.

Тем не менее, с отражением вы можете сделать экстраординарное, даже доступ к protected а также private Члены вне класса!

Ну что такое отражение?

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

преамбула

У нас есть класс с именем Grandpas и скажем, у нас есть три свойства. Для простоты рассмотрим три дедушки с именами:

  • Марк Генри
  • Джон Клаш
  • Уилл Джонс

Давайте сделаем их (назначим модификаторы) public, protected а также private соответственно. Вы очень хорошо знаете, что protected а также private члены не могут быть доступны за пределами класса. Теперь давайте противоречим утверждению, используя отражение.

Код

<?php

class GrandPas   // The Grandfather's class
{
    public     $name1 = 'Mark Henry';  // This grandpa is mapped to a public modifier
    protected  $name2 = 'John Clash';  // This grandpa is mapped to a protected  modifier
    private    $name3 = 'Will Jones';  // This grandpa is mapped to a private modifier
}


# Scenario 1: without reflection
$granpaWithoutReflection = new GrandPas;

# Normal looping to print all the members of this class
echo "#Scenario 1: Without reflection<br>";
echo "Printing members the usual way.. (without reflection)<br>";
foreach($granpaWithoutReflection as $k=>$v)
{
    echo "The name of grandpa is $v and he resides in the variable $k<br>";
}

echo "<br>";

#Scenario 2: Using reflection

$granpa = new ReflectionClass('GrandPas'); // Pass the Grandpas class as the input for the Reflection class
$granpaNames=$granpa->getDefaultProperties(); // Gets all the properties of the Grandpas class (Even though it is a protected or private)


echo "#Scenario 2: With reflection<br>";
echo "Printing members the 'reflect' way..<br>";

foreach($granpaNames as $k=>$v)
{
    echo "The name of grandpa is $v and he resides in the variable $k<br>";
}

Выход:

#Scenario 1: Without reflection
Printing members the usual way.. (Without reflection)
The name of grandpa is Mark Henry and he resides in the variable name1

#Scenario 2: With reflection
Printing members the 'reflect' way..
The name of grandpa is Mark Henry and he resides in the variable name1
The name of grandpa is John Clash and he resides in the variable name2
The name of grandpa is Will Jones and he resides in the variable name3

Распространенные заблуждения:

Пожалуйста, не путайте с приведенным ниже примером. Как вы все еще можете видеть, private а также protected члены не могут быть доступны за пределами класса без использования отражения

<?php

class GrandPas   // The Grandfather's class
{
    public     $name1 = 'Mark Henry';  // This grandpa is mapped to a public modifier
    protected  $name2 = 'John Clash';  // This grandpa is mapped to a protected modifier
    private    $name3 = 'Will Jones';  // This grandpa is mapped to a private modifier
}

$granpaWithoutReflections = new GrandPas;
print_r($granpaWithoutReflections);

Выход:

GrandPas Object
(
    [name1] => Mark Henry
    [name2:protected] => John Clash
    [name3:GrandPas:private] => Will Jones
)

Функции отладки

print_r, var_export а также var_dump являются функциями отладчика. Они представляют информацию о переменной в удобочитаемой форме. Эти три функции покажут protected а также private свойства объектов с PHP 5. Статические члены класса не будут показаны.


Больше ресурсов:


private - может быть доступен только из класса WITHIN

защищенный - может быть доступен из класса и наследования классов

public - можно получить доступ из кода вне класса, а также

Это относится как к функциям, так и к переменным.

Обычно считается, что по умолчанию рекомендуется использовать минимальную видимость, поскольку это способствует инкапсуляции данных и хорошему дизайну интерфейса. Рассматривая переменную члена и видимость метода, подумайте о роли, которую член играет во взаимодействии с другими объектами.

Если вы "кодируете интерфейс, а не реализацию", то, как правило, довольно просто принимать решения о видимости. В общем случае переменные должны быть закрытыми или защищенными, если у вас нет веских причин для их раскрытия. Вместо этого используйте общедоступные методы доступа (методы получения / установки), чтобы ограничить и регулировать доступ к внутренним элементам класса.

Чтобы использовать автомобиль в качестве аналогии, такие вещи, как скорость, передача и направление, будут частными переменными экземпляра. Вы не хотите, чтобы водитель непосредственно манипулировал такими вещами, как соотношение воздух / топливо. Вместо этого вы выставляете ограниченное количество действий в качестве открытых методов. Интерфейс для автомобиля может включать такие методы, как accelerate(), deccelerate()/brake(), setGear(), turnLeft(), turnRight(), так далее.

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

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

Изменить: так как кажется, что вы все еще находитесь в процессе изучения объектно-ориентированных концепций (которые намного сложнее освоить, чем синтаксис любого языка), я настоятельно рекомендую Мэтта Зандстра взять копию PHP Objects, Patterns и Practice. Эта книга впервые научила меня, как эффективно использовать ООП, а не просто научила меня синтаксису. Я выучил синтаксис за годы до этого, но это было бесполезно без понимания "почему" ООП.

Разница заключается в следующем:

Public:: Публичная переменная или метод могут быть доступны непосредственно любому пользователю класса.

Protected:: Защищенная переменная или метод не могут быть доступны пользователям класса, но могут быть доступны внутри подкласса, который наследуется от класса.

Private:: Доступ к закрытой переменной или методу возможен только из класса, в котором он определен. Это означает, что закрытая переменная или метод не могут быть вызваны из дочернего элемента, который расширяет класс.

Области видимости с абстрактными примерами:: Облегчает понимание

Эта видимость свойства или метода определяется предварительным исправлением объявления одного из трех ключевых слов (Public, protected и private).

Открытый: если свойство или метод определен как открытый, это означает, что к нему можно обращаться как к объекту, так и к нему.

  • Аннотация например Подумайте о публичной видимости как о "публичном пикнике", к которому может прийти каждый.

Защищенный: когда видимость свойства или метода установлена ​​для защищенных членов, доступ может быть только внутри самого класса и унаследованных и наследующих классов. (Унаследовано: - класс может иметь все свойства и методы другого класса).

  • Думайте как защищенная область видимости как "Пикник компании", где членам компании и членам их семей разрешено не публиковаться. Это наиболее распространенное ограничение объема.

Личный: когда видимость свойства или метода установлена ​​на частный, только класс, имеющий закрытые члены, может получить доступ к этим методам и свойствам (внутри класса), независимо от того, какое отношение класса там существует.

  • с аналогией с пикником думайте как "пикник компании, где разрешены только члены компании" на пикнике. не семья и широкая публика не допускаются.
/**
 * Define MyClass
 */
class MyClass
{
    public $public = 'Public';
    protected $protected = 'Protected';
    private $private = 'Private';

    function printHello()
    {
        echo $this->public;
        echo $this->protected;
        echo $this->private;
    }
}

$obj = new MyClass();
echo $obj->public; // Works
echo $obj->protected; // Fatal Error
echo $obj->private; // Fatal Error
$obj->printHello(); // Shows Public, Protected and Private


/**
 * Define MyClass2
 */
class MyClass2 extends MyClass
{
    // We can redeclare the public and protected method, but not private
    protected $protected = 'Protected2';

    function printHello()
    {
        echo $this->public;
        echo $this->protected;
        echo $this->private;
    }
}

$obj2 = new MyClass2();
echo $obj2->public; // Works
echo $obj2->private; // Undefined
echo $obj2->protected; // Fatal Error
$obj2->printHello(); // Shows Public, Protected2, Undefined

Извлеченный из:

http://php.net/manual/en/language.oop5.visibility.php

⚡️ Вот простой способ запомнить объем public, protected а также private,

PUBLIC:

  • public область действия: открытая переменная / функция доступна как для объектов, так и для других классов.

PROTECTED:

  • protected Область действия: защищенная переменная / функция доступна для всех классов, расширяющих текущий класс.
  • Нет! Объекты не могут получить доступ к этой области

PRIVATE:

  • private область действия: закрытая переменная / функция видна только в текущем классе, где она определяется.
  • Нет! Класс, расширяющий текущий класс, не может получить доступ к этой области.
  • Нет! Объекты не могут получить доступ к этой области.

Прочитайте видимость метода или переменной в руководстве по PHP.

Учитывая "когда":
Я склонен объявлять все как частное изначально, если я не совсем уверен. Причина в том, что обычно сделать публичный метод общедоступным гораздо проще, чем наоборот. Это потому, что вы можете быть уверены, что приватный метод нигде не использовался, кроме как в самом классе. Публичный метод уже может использоваться везде, возможно, требующий обширной переписывания.

Обновление: я иду по умолчанию protected в настоящее время, потому что я пришел к выводу, что он достаточно хорош для инкапсуляции и не мешает расширению классов (чего я в любом случае стараюсь избегать). Только если у меня будет веская причина использовать два других, я сделаю это.

Хорошая причина для private Метод будет тем, который реализует что-то, присущее этому объекту, что даже расширяющийся класс не должен изменяться (фактическая причина, в дополнение к инкапсуляции, как внутреннее управление состоянием). В конце концов, все еще достаточно легко отследить, где protected Метод используется обычно, поэтому я по умолчанию protected Настоящее время. Может быть, не 100% объективный опыт "в окопах", я признаю.

Для меня это самый полезный способ понять три типа свойств:

Public: используйте это, когда у вас все в порядке с прямой доступ к этой переменной и ее изменение из любой точки вашего кода.

Пример использования извне класса:

$myObject = new MyObject()
$myObject->publicVar = 'newvalue';
$pubVar = $myObject->publicVar;

Защищено: Используйте это, когда вы хотите заставить других программистов (и вас самих) использовать методы получения / установки вне класса при доступе и установке переменных (но вы должны быть последовательными и использовать методы получения и установки также внутри класса). Это или private как правило, по умолчанию вы должны установить все свойства класса.

Зачем? Потому что, если вы решите в какой-то момент в будущем (возможно, даже через 5 минут), что вы хотите манипулировать значением, возвращаемым для этого свойства, или делать что-то с ним перед получением / установкой, вы можете сделать это без рефакторинга везде, где есть использовал его в своем проекте.

Пример использования извне класса:

$myObject = new MyObject()
$myObject->setProtectedVar('newvalue');
$protectedVar = $myObject->getProtectedVar();

Частный: private свойства очень похожи на protected свойства. Но отличительная особенность / отличие заключается в том, что private также делает его недоступным для дочерних классов без использования метода получения или установки родительского класса.

Таким образом, в основном, если вы используете методы получения и установки для свойства (или если оно используется только внутри родительского класса и не предназначено для того, чтобы быть доступным где-либо еще), вы также можете сделать это privateпросто чтобы никто не пытался использовать его напрямую и вводить ошибки.

Пример использования внутри дочернего класса (который расширяет MyObject):

$this->setPrivateVar('newvalue');
$privateVar = $this->getPrivateVar();

Возрождаем старый вопрос, но я думаю, что действительно хороший способ подумать об этом с точки зрения API, который вы определяете.

  • public - Все, что помечено как public, является частью API, который любой, кто использует ваш класс / интерфейс / другой, будет использовать и полагаться на него.

  • protected - Не обманывайте себя, это тоже часть API! Люди могут создавать подклассы, расширять ваш код и использовать все, что помечено как защищенное.

  • private - Частные свойства и методы могут быть изменены так, как вам нравится. Никто другой не может использовать их. Это единственное, что вы можете изменить, не внося серьезных изменений.

Или в терминах Semver:

  • Изменения ни к чему public или же protected следует учитывать ОСНОВНЫЕ изменения.

  • Что-нибудь новое public или же protected должно быть (по крайней мере) незначительным

  • Только новые / изменения ни к чему private может быть патч

Так что с точки зрения поддержки кода, хорошо быть осторожным в том, что вы делаете public или же protected потому что это то, что вы обещаете своим пользователям.

Руководство по PHP хорошо прочитано по этому вопросу здесь.

Видимость свойства или метода может быть определена путем добавления префикса объявления к ключевым словам public, protected или private. Члены класса, объявленные публичными, могут быть доступны везде. Члены, объявленные защищенными, могут быть доступны только внутри самого класса и унаследованных и родительских классов. Члены, объявленные как частные, могут быть доступны только классу, который определяет член.

Они там, чтобы учесть разные уровни инкапсуляции

Переменные в PHP преобразуются в три разных типа:

Public: значения этих типов переменных доступны во всей области видимости и требуют выполнения вашего кода. объявить как: public $examTimeTable;

Private: значения этого типа переменных доступны только для класса, к которому он принадлежит. private $classRoomComputers;

Защищено: Значения этого класса только и доступны только тогда, когда доступ был предоставлен в форме наследования или их дочернего класса. обычно используется :: предоставить доступ по родительскому классу

protected $familyWealth;

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

private - только текущий класс будет иметь доступ к полю или методу.

protected - только текущий класс и подклассы (а иногда и классы того же пакета) этого класса будут иметь доступ к полю или методу.

public - любой класс может ссылаться на поле или вызывать метод.

Когда мы следуем объектно-ориентированному php в нашем проекте, мы должны следовать некоторым правилам, чтобы использовать модификаторы доступа в php. Сегодня мы собираемся ясно узнать, что такое модификатор доступа и как мы можем его использовать. Модификаторы доступа PHP используются для установки прав доступа с классами PHP и их членами, которые являются функциями и переменными, определенными в области видимости класса. В php есть три области для учеников.

  1. ОБЩЕСТВЕННОЕ
  2. ЧАСТНЫЙ
  3. ЗАЩИЩЕНЫ

Теперь давайте посмотрим на следующее изображение, чтобы понять уровень доступа модификатора доступа

Теперь давайте взглянем на следующий список, чтобы узнать о возможных ключевых словах PHP, используемых в качестве модификаторов доступа.

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

private: - члены класса с этим ключевым словом будут доступны внутри самого класса. мы не можем получить доступ к личным данным из подкласса. Он защищает членов от доступа извне класса.

protected: - то же самое как private, за исключением того, что позволяет подклассам получать доступ к защищенным членам суперкласса.

Теперь посмотрите таблицу, чтобы понять модификатор доступа. Читать статью php access modifire.

Public: это состояние по умолчанию, когда вы объявляете переменную или метод, доступ к чему-либо непосредственно к объекту.

Защищено: Доступ к нему возможен только внутри объекта и подклассов.

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

Три уровня видимости

В ООП PHP у нас есть три уровня видимости для свойств и методов класса: открытый, защищенный и закрытый. Видимость объявляется с использованием ключевого слова видимости, чтобы объявить, какой уровень видимости имеет свойство или метод. Три уровня определяют, можно ли получить доступ к свойству или методу вне класса и в классах, которые расширяют класс.

общественного

Этот уровень не имеет ограничений, поэтому его можно вызывать в любой области видимости. Это означает, что открытое свойство объекта может быть извлечено и изменено из любого места в программе - в классе, подклассе или извне класса.

защищенный

Доступ к защищенным свойствам и методам можно получить из класса, в котором они объявлены, или из любого класса, который их расширяет. К ним нельзя получить доступ снаружи класса или подкласса.

Частный

Частное свойство или метод не могут быть доступны для подкласса класса, в котором они определены. Если у вас есть класс с защищенным свойством и частным свойством, а затем вы можете расширить этот класс в подклассе, вы можете получить доступ к защищенному свойству, но не частная собственность.

Другие вопросы по тегам