什么时候以及为什么我应该在类中使用公共、私有和受保护的函数和变量?它们之间的区别是什么?

例子:

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

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

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

当前回答

公众:

当你将一个方法(函数)或属性(变量)声明为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'

保护:

当将方法(函数)或属性(变量)声明为受保护时,可以通过

声明它的类。 继承上述声明类的类。

外部成员不能访问这些变量。“局外人”的意思是它们不是所声明类本身的对象实例。

例子:

<?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


私人:

当你将一个方法(函数)或属性(变量)声明为私有时,这些方法和属性可以通过以下方式访问:

声明它的类。

外部成员不能访问这些变量。在某种意义上,它们不是已声明类本身的对象实例,甚至不是继承已声明类的类。

例子:

<?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

准确的错误消息将是:

注意:Undefined属性:Daddy::$name 致命错误:无法访问私有属性GrandPa::$name


使用反射解剖爷爷类

这个主题并没有超出范围,我在这里添加它只是为了证明反射是非常强大的。正如我在上面三个例子中所述,受保护的和私有的成员(属性和方法)不能在类之外被访问。

然而,有了反射,你甚至可以在类外部访问受保护的和私有的成员。

什么是反射?

反射增加了逆向工程类、接口、 函数、方法和扩展。此外,他们还提供了一些方法 检索函数、类和方法的文档注释。

序言

我们有一个名为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是调试器函数。它们以人类可读的形式呈现关于变量的信息。这三个函数将显示PHP 5中对象的受保护属性和私有属性。静态类成员将不会显示。


更多资源:

PHP手册- OOP属性 PHP手册- OOP可见性 PHP类的可见性 Jordizle.com - Public, Private,在PHP 5中受保护


其他回答

通常认为最佳实践是默认为所需的最低可见性,因为这可以促进数据封装和良好的接口设计。在考虑成员变量和方法可见性时,要考虑成员在与其他对象的交互中所扮演的角色。

如果你“为接口而不是实现编码”,那么做出可见性决策通常是相当简单的。一般来说,变量应该是私有的或受保护的,除非您有充分的理由公开它们。使用公共访问器(getter /setter)来限制和规范对类内部的访问。

以汽车为例,速度、档位和方向等都是私有实例变量。你不希望司机直接操纵像空气/燃料比这样的事情。相反,您将有限数量的操作公开为公共方法。汽车的接口可能包括加速()、减速()/刹车()、setGear()、turnLeft()、turnRight()等方法。

司机不知道也不应该关心这些动作是如何由汽车内部实现的,暴露这些功能可能会对司机和路上的其他人造成危险。因此,设计一个公共接口并将数据封装在该接口后面是一个很好的实践。

这种方法还允许您更改和改进类中公共方法的实现,而不会破坏接口与客户端代码的约定。例如,你可以改进accelerate()方法,使其更省油,但该方法的使用将保持不变;客户端代码不需要更改,但仍然可以获得效率提高的好处。

编辑:既然你似乎还在学习面向对象的概念(它比任何语言的语法都更难掌握),我强烈建议你去读Matt Zandstra的《PHP对象、模式和实践》。这是第一本教会我如何有效地使用OOP的书,而不仅仅是教我语法。我在几年前就已经学过语法了,但如果不理解OOP的“原因”,这是没有用的。

差异如下:

Public:类的任何用户都可以直接访问公共变量或方法。

Protected::受保护的变量或方法不能被类的用户访问,但可以在继承自该类的子类中访问。

Private::私有变量或方法只能从定义它的类内部访问。这意味着不能从扩展该类的子类调用私有变量或方法。

private -只能从类内部访问

protected -可以从类内部和继承类中访问

public -也可以从类外部的代码访问

这不仅适用于变量,也适用于函数。

对我来说,这是理解这三种属性类型最有用的方法:

Public:当您可以在代码中的任何地方直接访问和更改此变量时,使用此选项。

类外部的示例用法:

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

Protected:当你想强迫其他程序员(和你自己)在访问和设置变量时在类外部使用getter /setter时使用这个选项(但你应该保持一致,在类内部也使用getter和setter)。This或private往往是设置所有类属性的默认方式。

为什么?因为如果你决定在未来的某个时候(可能甚至在5分钟内),你想要操作该属性返回的值,或者在获取/设置之前对它做一些事情,你可以这样做,而无需重构你在项目中使用它的所有地方。

类外部的示例用法:

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

Private:私有属性与受保护属性非常相似。但显著的特点/区别是,将其设为私有也使得子类无法访问它,而无需使用父类的getter或setter。

因此,基本上,如果你正在为一个属性使用getter和setter(或者如果它只在父类内部使用,并且它并不意味着在其他任何地方都可以访问),你最好将它设置为私有,只是为了防止任何人试图直接使用它并引入错误。

在子类(扩展MyObject)中的使用示例:

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

你使用:

公共作用域,使该属性/方法可从任何地方、对象的其他类和实例使用。 当您希望您的属性/方法仅在其自己的类中可见时,则为私有作用域。 当您希望使您的属性/方法在扩展当前类(包括父类)的所有类中可见时,使用Protected作用域。

如果你不使用任何可见性修饰符,属性/方法将是公共的。

更多信息:(综合信息)

PHP手册-可见性