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


989

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

Примеры:

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

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

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

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

70
Я действительно считаю, что этот вопрос должен быть публичным, а не защищенным.
dotancohen

Ответы:


1261

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

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

  • private область, когда вы хотите, чтобы ваше свойство / метод было видно только в его собственном классе.

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

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


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

4
@ Шахид - я не понимаю твою точку зрения. Любой класс, который расширяет класс A, также расширяет родительский класс A, нет?
JDelage

4
@JDelage - Пожалуйста, смотрите ссылкуhttp://www.php.net/manual/en/language.oop5.visibility.php#109324
Шахид

4
@Growler Зачем вообще использовать объекты?
J.Steve

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

1176

дд

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

Когда вы объявляете метод (функцию) или свойство (переменную) как 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

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

Примечание: неопределенное свойство: Daddy :: $ 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. Статический класс не будет показано.


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



извинения за опоздание добавить на этот конво. Можете ли вы сказать мне, почему кто-то будет использовать их? Вы прекрасно объяснили, как они работают и т.д. Я просто хотел бы узнать о преимуществах использования каждого из них. Спасибо
JamesG

@JamesG это немного объяснено в другом комментарии выше. stackoverflow.com/questions/4361553/…
cjmling

Я не знаю, почему, может быть, это немного выходит из этого вопроса, но никто не упомянул, что в PHP есть еще два модификатора доступа: abstract и final, это ключевое слово может использоваться только для классов PHP, но оно все еще имеет доступ к модификаторам
bxN5

1
Я бы посоветовал вам прочитать объяснение об абстракции, предоставленное Дайрией Лахерой здесь: stackoverflow.com/questions/2558559/… . Это прекрасное дополнение к объяснениям Шанкара Дамодарана.
Хулио Марчи

83

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

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

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

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

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

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


3
Книга, рекомендованная в редакции этого поста, действительно очень хороша. Кусок, который я до сих пор оказался весьма поучительным. Первые несколько глав ответили на большинство моих вопросов, связанных с классом.
Иосия

Книги, которые позволили мне по-настоящему понять объекты, не загоняя свои мысли ненужными подробностями, как, например, примеры в Smalltalk, были написаны Дэвидом Тейлором, « Объектно-ориентированные технологии: руководство менеджера и бизнес-инжиниринг с объектной технологией» . Оба - всего 100 страниц, и каждый из них достаточно прост для чтения в полдень. Конечно, есть Gamma et al. Design Patterns , хотя основной подход можно просто описать как «подкласс того, что вы хотите изменить».
Патанджали

Очень хорошая аналогия. У вас есть один для защищенного против частного?
Янис Элмерис

79

private - можно получить доступ только из класса

protected - можно получить доступ из класса ВНУТРЕННЕГО класса

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

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


Не уверен, что защищенное определение здесь правильно, из фактически выбранного ответа кажется, Защищенный - Доступ к нему возможен только из унаследованного класса и далее, а не из исходного / родительского класса. Сказать «В классе» может быть немного странно.
pal4life

7
Я так не думаю, на самом деле кажется, что выбранный ответ - тот, который здесь сбивает с толку. Смотрите комментарий шахидов. ИМХО, защищенный метод очень хорошо доступен из исходного класса.
Олаф

может ли класс получить доступ к общедоступному классу другого класса?
Serjas

1
@Serjas: Нет, только другой объект, если это не статические методы / поля.
DanMan

Я не знаю, относится ли это ко всем языкам программирования, но в PHP «защищенные» свойства / методы могут быть доступны либо в классе, в котором он был объявлен, либо в классах, которые наследуются от класса, определяющего свойство / метод.
Джон Слегерс

25

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

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

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

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


17

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

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

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

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

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

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

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

  • с аналогией с пикником думайте как «пикник компании, где разрешены только члены компании» на пикнике. не семья и широкая публика не допускаются.

15
/**
 * 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


12

⚡️ Вот простой способ запомнить сферу public, protectedи private.

PUBLIC:

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

PROTECTED:

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

PRIVATE:

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

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


9

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

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

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


3
С помощью вашего обновления: Можете ли вы прояснить, как «достаточно хороший» и «хороший повод» сочетаются здесь? Например, использование privateбудет по-прежнему «достаточно хорошим» для использования, но вы не предлагаете, чтобы, хотя и более ранние причины, которые вы приводили, звучат как «хорошая причина»: инкапсуляция.
13

@hakre: причина, по которой мы должны стремиться к инкапсуляции, состоит в том, чтобы избежать утечки состояния во внешнюю область. protectedделает это уже, но вы сохраняете его гибким для расширения / наследования. Опять же, если у вас нет веских причин сделать это private.
DanMan

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

Я не буду спорить с вашей общей точкой зрения, потому что это достаточно справедливо, но protectedне проникает во внешнюю область (код, который вызывает / обращается к методу / полю), а только во внутреннюю область (расширяющие классы). Есть разница, настолько незначительная, насколько это возможно для вас. Гораздо проще отследить использование protectedполя, чем publicодного.
DanMan

6

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

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


6

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

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();


4

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

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

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

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

protected $familyWealth;


3

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

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

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

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

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

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

  • Что-нибудь новое publicили protectedдолжно быть (по крайней мере) МЕНЬШЕ

  • privateПАТЧ может быть только новый / изменения чего-либо

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


1

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

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

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

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

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

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

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

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


1

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

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

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


0

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

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

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

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

Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.