继承会拥有父类的 Private 属性和方法吗?

LemonLone
  • 237

看了PHP官方手册,也是“言简意赅”... 属性定义也直接是 var,这阿斗也是醉了,唉...

这里有个帖子,附上了测试的代码:http://bbs.phpchina.com/thread-116668-1-1.html

那到底继承会不会拥有父类的 private 属性和方法呢?继承在意义上来说是不是相当于父类简单的代码拷贝?是不是说子类也会有父类的 private,只是父类的 private 只能父类自己的方法访问,子类的 private 只能子类自己的方法访问?其实各自的 private 还是各自的?

回复
阅读 13.5k
6 个回答
沙渺
  • 21.8k
✓ 已被采纳

必须明确:private, publicprotected这几个关键字,和继承一毛钱的关系都没有。人家叫做“可见性(Visibility)”。基类不能控制自己的所有成员都被子类继承走,但可以约束某些被继承的的成员,对子类新增补的成员不可见

从“拥有”这个角度来说,子类是基类的套壳,基类的一切当然都在子类中“存在”——只是从子类的角度能不能“看得到”而已。

如下图(蓝色箭头部分),private member肯定存在在子类的存储空间中,也就是说肯定被子类继承到了。但子类新增的方法就是访问不到。

可访问性示例

这个事情从架构设计这个意义上,其实是很容易说清楚的。一个子类必须遵守基类的所有特征,然后在其上做出增补和扩充。而如果基类的所有成员都开放给子类随便看、随便改,基类的本来特征可以随便拆掉,那么还要继承做什么呢?

在实际工程中,基类的私有成员,要么是基类自己内部使用,要么是像上图(红色箭头部分)一样,通过protected方法做一个访问器,保证既把访问权间接开放给子类,又能在子类访问时执行一些必要的控制逻辑。

10xjzheng
  • 11

可以参考此链接访问控制(可见性),在此贴上部分讲解此问题的文字。关键部分已加粗

对属性或方法的访问控制,是通过在前面添加关键字 public(公有),protected(受保护)或 private(私有)来实现的。
被定义为公有的类成员可以在任何地方被访问。被定义为受保护的类成员则可以被其自身以及其子类和父类访问。被定义为私有的类成员则只能被其定义所在的类访问。

Example #1 属性声明

<?php
/**
 * 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; // 这行能被正常执行
echo $obj->protected; // 这行会产生一个致命错误
echo $obj->private; // 这行也会产生一个致命错误
$obj->printHello(); // 输出 Public、Protected 和 Private


/**
 * Define MyClass2
 */
class MyClass2 extends MyClass
{
    // 可以对 public 和 protected 进行重定义,但 private 而不能
    protected $protected = 'Protected2';

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

$obj2 = new MyClass2();
echo $obj2->public; // 这行能被正常执行
echo $obj2->private; // 未定义 private
echo $obj2->protected; // 这行会产生一个致命错误
$obj2->printHello(); // 输出 Public、Protected2 和 Undefined

?>

Example #2 方法声明

<?php
/**
 * Define MyClass
 */
class MyClass
{
    // 声明一个公有的构造函数
    public function __construct() { }

    // 声明一个公有的方法
    public function MyPublic() { }

    // 声明一个受保护的方法
    protected function MyProtected() { }

    // 声明一个私有的方法
    private function MyPrivate() { }

    // 此方法为公有
    function Foo()
    {
        $this->MyPublic();
        $this->MyProtected();
        $this->MyPrivate();
    }
}

$myclass = new MyClass;
$myclass->MyPublic(); // 这行能被正常执行
$myclass->MyProtected(); // 这行会产生一个致命错误
$myclass->MyPrivate(); // 这行会产生一个致命错误
$myclass->Foo(); // 公有,受保护,私有都可以执行


/**
 * Define MyClass2
 */
class MyClass2 extends MyClass
{
    // 此方法为公有
    function Foo2()
    {
        $this->MyPublic();
        $this->MyProtected();
        $this->MyPrivate(); // 这行会产生一个致命错误
    }
}

$myclass2 = new MyClass2;
$myclass2->MyPublic(); // 这行能被正常执行
$myclass2->Foo2(); // 公有的和受保护的都可执行,但私有的不行

class Bar 
{
    public function test() {
        $this->testPrivate();
        $this->testPublic();
    }

    public function testPublic() {
        echo "Bar::testPublic\n";
    }
    
    private function testPrivate() {
        echo "Bar::testPrivate\n";
    }
}

class Foo extends Bar 
{
    public function testPublic() {
        echo "Foo::testPublic\n";
    }
    
    private function testPrivate() {
        echo "Foo::testPrivate\n";
    }
}

$myFoo = new foo();
$myFoo->test(); // Bar::testPrivate 
                // Foo::testPublic
?>
做向上的少年
  • 13
<?php
class baseClass{

    private $name;

    function __set($key,$value){
        $this->$key = $value;
        echo 'base.set:'.$this->$key.'<br>';
    }

    function __get($param){
        return 'base.get:'.$this->$param;
    }
}
class sonClass extends baseClass{
    private $email;

    function __set($key,$value){
        $this->$key = $value;
        echo 'son.set:'.$this->$key.'<br>';
    }

    function __get($param){
        return 'son.get:'.$this->$param;
    }
}

$son = new sonClass();

$base = new baseClass();
print_r(get_object_vars($base));

print_r(get_object_vars($son));


$son->name = '5';

print_r(get_object_vars($base));

print_r(get_object_vars($son));

echo $son->name;
echo $base->name;
?>

运行上面的代码你就会发现,最初的时候属性未赋值前打印两个实例化的类的所有属性值,对比给属性赋值之后再次打印的两个类的属性值,我们会发现,其实sonClass并没有继承baseClassprivate name属性,而对于类(sonClass)中未定义的属性($name),php会默认自动创建,从输出结果来看,就会发现 $son->name = '5'; 这条语句,其实走的sonClass中的__set()函数,而并非baseClass中的set()函数,也即是sonClass没有继承baseClassprivate属性。

所以,“子类继承只会继承publicprotected定义的方法属性这句话,并没有错”

<?php
class baseClass{

    private $name;

    function __set($key,$value){
        $this->$key = $value;
        echo 'base.set:'.$this->$key.'<br>';
    }

    function __get($param){
        return 'base.get:'.$this->$param;
    }
}
class sonClass extends baseClass{
    private $email;

    // function __set($key,$value){
    //     $this->$key = $value;
    //     echo 'son.set:'.$this->$key.'<br>';
    // }

    // function __get($param){
    //     return 'son.get:'.$this->$param;
    // }
}

$son = new sonClass();
$base = new baseClass();


print_r(get_object_vars($base));
echo '<br />';

print_r(get_object_vars($son));
echo '<hr />';


$son->name = '5';

print_r(get_object_vars($base));
echo '<br />';
print_r(get_object_vars($son));
echo '<hr />';

echo $son->name;
echo '<hr />';
echo $base->name;
?>

clipboard.png
这说明父类的私有变量已被子类继承

当然不会,后面接力。

我感觉继承 应该是向上查找个一个过程, 当我们需要访问一个类中的属性, 首先会看当前类有没有, 若没有就到它的父类中查询! var_dump()打印一个子类的时候是可以看到父类的私有属性, 但是不能访问

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
宣传栏