看了PHP官方手册,也是“言简意赅”... 属性定义也直接是 var,这阿斗也是醉了,唉...
这里有个帖子,附上了测试的代码:http://bbs.phpchina.com/thread-116668-1-1.html
那到底继承会不会拥有父类的 private 属性和方法呢?继承在意义上来说是不是相当于父类简单的代码拷贝?是不是说子类也会有父类的 private,只是父类的 private 只能父类自己的方法访问,子类的 private 只能子类自己的方法访问?其实各自的 private 还是各自的?
我感觉继承 应该是向上查找个一个过程, 当我们需要访问一个类中的属性, 首先会看当前类有没有, 若没有就到它的父类中查询! var_dump()打印一个子类的时候是可以看到父类的私有属性, 但是不能访问
必须明确:private
, public
和protected
这几个关键字,和继承一毛钱的关系都没有。人家叫做“可见性(Visibility)”。基类不能控制自己的所有成员都被子类继承走,但可以约束某些被继承的的成员,对子类新增补的成员不可见。
从“拥有”这个角度来说,子类是基类的套壳,基类的一切当然都在子类中“存在”——只是从子类的角度能不能“看得到”而已。
如下图(蓝色箭头部分),private member肯定存在在子类的存储空间中,也就是说肯定被子类继承到了。但子类新增的方法就是访问不到。
这个事情从架构设计这个意义上,其实是很容易说清楚的。一个子类必须遵守基类的所有特征,然后在其上做出增补和扩充。而如果基类的所有成员都开放给子类随便看、随便改,基类的本来特征可以随便拆掉,那么还要继承做什么呢?
在实际工程中,基类的私有成员,要么是基类自己内部使用,要么是像上图(红色箭头部分)一样,通过protected
方法做一个访问器,保证既把访问权间接开放给子类,又能在子类访问时执行一些必要的控制逻辑。
当然不会,后面接力。
<?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
并没有继承baseClass
的private name
属性,而对于类(sonClass
)中未定义的属性($name
),php会默认自动创建,从输出结果来看,就会发现 $son->name = '5';
这条语句,其实走的sonClass
中的__set()
函数,而并非baseClass
中的set()
函数,也即是sonClass
没有继承baseClass
的private
属性。
所以,“子类继承只会继承public
和protected
定义的方法属性这句话,并没有错”
可以参考此链接访问控制(可见性),在此贴上部分讲解此问题的文字。关键部分已加粗
对属性或方法的访问控制,是通过在前面添加关键字 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
?>
<?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;
?>
这说明父类的私有变量已被子类继承