php面向对象(OOP)编程

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了php面向对象(OOP)编程相关的知识,希望对你有一定的参考价值。


大多数类都有一种称为构造函数的特殊方法。当创建一个对象时,它将自动调用构造函数,也就是使用new这个关键字来实例化对象的时候自动调用构造方法。构 造函数的声明与其它操作的声明一样,只是其名称必须是__construct( )。这是php5中的变化,以前的版本中,构造函数的名称必须与类名相同,这种在PHP5中仍然可以用,但现在以经很少有人用了,这样做的好处是可以使构 造函数独立于类名,当类名发生改变时不需要改相应的构造函数名称了。为了向下兼容,如果一个类中没有名为__construct( )的方法,PHP将搜索一个php4中的写法,与类名相同名的构造方法。

格式:function __construct ( [参数] ) ... ...

在一个类中只能声明一个构造方法,而是只有在每次创建对象的时候都会去调用一次构造方法,不能主动的调用这个方法,所以通常用它执行一些有用的初始化任务。比如对成属性在创建对象的时候赋初值。

析构函数:

析构函数不能带有任何参数。

格式:function __destruct ( ) ... ...

​​<?​​       


​​// 创建一个人类​​


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​var​​ ​​ $name​​ ​​; ​​ ​​// 人的名子​​


​​var​​ ​​ $sex​​ ​​; ​​ ​​// 人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​// 人的年龄​​





​​// 定义一个构造方法参数为姓名$name、性别$sex和年龄$age​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​) ​​


​​// 通过构造方法传进来的$name给成员属性$this->name赋初使值​​


​​$this​​ ​​->name = ​​ ​​$name​​ ​​;​​





​​// 通过构造方法传进来的$sex给成员属性$this->sex赋初使值​​


​​$this​​ ​​->sex = ​​ ​​$sex​​ ​​;​​





​​// 通过构造方法传进来的$age给成员属性$this->age赋初使值​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​​​





​​// 这个人的说话方法​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​


​​​​





​​// 这是一个析构函数,在对象销毁前调用​​


​​function​​ ​​ __destruct() ​​


​​echo​​ ​​ "再见"​​ ​​ . ​​ ​​$this​​ ​​->name;​​


​​​​


​​​​





​​// 通过构造方法创建3个对象$p1、p2、$p3,分别传入三个不同的实参为姓名、性别和年龄​​


​​$p1​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"张三"​​ ​​, ​​ ​​"男"​​ ​​, 20);​​


​​$p2​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"李四"​​ ​​, ​​ ​​"女"​​ ​​, 30);​​


​​$p3​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"王五"​​ ​​, ​​ ​​"男"​​ ​​, 40);​​





​​// 下面访问$p1对象中的说话方法​​


​​$p1​​ ​​->say();​​





​​// 下面访问$p2对象中的说话方法​​


​​$p2​​ ​​->say();​​





​​// 下面访问$p3对象中的说话方法​​


​​$p3​​ ​​->say();​​


​​?>​​


输出结果为:

我的名子叫:张三 性别:男 我的年龄是:20我的名子叫:李四 性别:女 我的年龄是:30我的名子叫:王五 性别:男 我的年龄是:40
再见王五
再见李四
再见张三

 注意:

堆栈的形式放在内存中,所以最后调用 析构函数 的时候,输出顺序是按 后进先出

使用private这个关键字来对属性和方法进行封装:

原来的成员:

​​var​​         ​​ $name​​        ​​;  ​​        ​​// 声明人的姓名​​       


​​var​​ ​​ $sex​​ ​​; ​​ ​​// 声明人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​// 声明人的年龄​​


​​function​​ ​​ run()……​​


改成封装的形式:

​​private​​         ​​ $name​​        ​​;  ​​        ​​// 把人的姓名使用private关键字进行封装​​       


​​private​​ ​​ $sex​​ ​​; ​​ ​​// 把人的性别使用private关键字进行封装​​


​​private​​ ​​ $age​​ ​​; ​​ ​​// 把人的年龄使用private关键字进行封装​​


​​private​​ ​​ function​​ ​​ run()…… ​​ ​​// 把人的走路方法使用private关键字进行封装​​


注意:只要是成员属性前面有其它的关键字就要去掉原有的关键字”var”。

通过private就可以把人的成员(成员属性和成员方法)封装上了。封装上的成员就不能被类外面直接访问了,只有对象内部自己可以访问,下面的代码会产生错误:

​​<?php​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​private​​ ​​ $name​​ ​​; ​​ ​​// 人的名子,被private封装上了​​


​​private​​ ​​ $sex​​ ​​; ​​ ​​// 人的性别, 被private封装上了​​


​​private​​ ​​ $age​​ ​​; ​​ ​​// 人的年龄, 被private封装上了​​





​​// 这个人可以说话的方法​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​


​​​​





​​// 这个人可以走路的方法, 被private封装上了​​


​​private​​ ​​ function​​ ​​ run() ​​


​​echo​​ ​​ "这个人在走路"​​ ​​;​​


​​​​


​​​​





​​// 实例化一个人的实例对象​​


​​$p1​​ ​​ = ​​ ​​new​​ ​​ Person();​​





​​// 试图去给私有的属性赋值, 结果会发生错误​​


​​$p1​​ ​​->name = ​​ ​​"张三"​​ ​​;​​


​​$p1​​ ​​->sex = ​​ ​​"男"​​ ​​;​​


​​$p1​​ ​​->age = 20;​​





​​// 试图去打印私有的属性, 结果会发生错误​​


​​echo​​ ​​ $p1​​ ​​->name;​​


​​echo​​ ​​ $p1​​ ​​->sex;​​


​​echo​​ ​​ $p1​​ ​​->age;​​





​​// 试图去打印私有的成员方法, 结果会发生错误​​


​​$p1​​ ​​->run();​​


​​?>​​


输出结果为:

Fatal error: Cannot access private property Person::$name
Fatal error: Cannot access private property Person::$sex
Fatal error: Cannot access private property Person::$age
Fatal error: Cannot access private property Person::$name
Fatal error: Call to private method Person::run() from context 

从上面的实例可以看到, 私有的成员是不能被外部访问的, 因为私有成员只能在本对象内部自己访问,比如,$p1这个对象自己想把他的私有属性说出去,在say()这个方法里面访问了私有属性,这样是可以。

没有加任何访问控制,默认的是public的,任何地方都可以访问。

​​// 这个人可以说话的方法, 说出自己的私有属性,在这里也可以访问私有方法​​       


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​





​​// 在这里也可以访问私有方法​​


​​//$this->run();​​


​​​​


因为成员方法say()是公有的, 所以我们在类的外部调用say()方法是可以的,改变上面的代码:

​​<?php​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​private​​ ​​ $name​​ ​​; ​​ ​​//人的名子,被private封装上了​​


​​private​​ ​​ $sex​​ ​​; ​​ ​​//人的性别, 被private封装上了​​


​​private​​ ​​ $age​​ ​​; ​​ ​​//人的年龄, 被private封装上了​​





​​// 定义一个构造方法参数为私有的属性姓名$name、性别$sex和年龄$age进行赋值​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​) ​​


​​// 通过构造方法传进来的$name给私有成员属性$this->name赋初使值​​


​​$this​​ ​​->name = ​​ ​​$name​​ ​​;​​





​​// 通过构造方法传进来的$sex给私有成员属性$this->sex赋初使值​​


​​$this​​ ​​->sex = ​​ ​​$sex​​ ​​;​​





​​// 通过构造方法传进来的$age给私有成员属性$this->age赋初使值​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​​​





​​// 这个人可以说话的方法, 说出自己的私有属性,在这里也可以访问私有方法​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​


​​​​


​​​​





​​// 通过构造方法创建3个对象$p1、p2、$p3,分别传入三个不同的实参为姓名、性别和年龄​​


​​$p1​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"张三"​​ ​​, ​​ ​​"男"​​ ​​, 20);​​


​​$p2​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"李四"​​ ​​, ​​ ​​"女"​​ ​​, 30);​​


​​$p3​​ ​​ = ​​ ​​new​​ ​​ Person(​​ ​​"王五"​​ ​​, ​​ ​​"男"​​ ​​, 40);​​





​​// 下面访问$p1对象中的说话方法​​


​​$p1​​ ​​->say();​​





​​// 下面访问$p2对象中的说话方法​​


​​$p2​​ ​​->say();​​





​​// 下面访问$p3对象中的说话方法​​


​​$p3​​ ​​->say();​​


​​?>​​


输出结果为:

我的名子叫:张三 性别:男 我的年龄是:20我的名子叫:李四 性别:女 我的年龄是:30我的名子叫:王五 性别:男 我的年龄是:40

因为构造方法是默认的公有方法(构造方法不要设置成私有的),所以在类的外面可以访问到,这样就可以使用构造方法创建对象, 另外构造方法也是类里面的函数,所以可以用构造方法给私有的属性赋初值。Say()的方法是默认公有的, 所以在外面也可以访问的到, 说出他自己的私有属性。

从上面的例子中我们可以看到, 私有的成员只能在类的内部使用, 不能被类外部直接来存取, 但是在类的内部是有权限访问的, 所以有时候我们需要在类的外面给私有属性赋值和读取出来,也就是给类的外部提供一些可以存取的接口,上例中构造方法就是一种赋值的形式, 但是构造方法只是在创建对象的时候赋值,如果我们已经有一个存在的对象了,想对这个存在的对象赋值, 这个时候,如果你还使用构造方法传值的形式传值, 那么就创建了一个新的对象,并不是这个已存在的对象了。所以我们要对私有的属性做一些可以被外部存取的接口,目的就是可以在对象存在的情况下,改变和存取 属性的值,但要注意,只有需要让外部改变的属性才这样做,不想让外面访问的属性是不做这样的接口的,这样就能达到封装的目的,所有的功能都是对象自己来完 成,给外面提供尽量少的操作。

如果给类外部提供接口,可以为私有属性在类外部提供设置方法和获取方法,来操作私有属性。例如:

​​prvate ​​        ​​$age​​        ​​; ​​        ​​// 私有的属性年龄​​       


​​function​​ ​​ setAge(​​ ​​$age​​ ​​) ​​ ​​// 为外部提供一个公有设置年龄的方法 ​​


​​if​​ ​​ (​​ ​​$age​​ ​​<0 || ​​ ​​$age​​ ​​>130) ​​ ​​// 在给属性赋值的时候,为了避免非法值设置给属性​​


​​return​​ ​​;​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​​​





​​function​​ ​​ getAge() ​​ ​​// 为外部提供一个公有获取年龄的方法 ​​


​​return​​ ​​(​​ ​​$this​​ ​​->age);​​


​​​​


上面的方法是为一个成员属性设置和获取值, 当然你也可以为每个属性用同样的方法对其进行赋值和取值的操作,完成在类外部的存取工作。



一般来说,总是把类的属性定义为private,这更符合现实的逻辑。但是, 对属性的读取和赋值操作是非常频繁的,因此在PHP5中,预定义了两个函数”__get()”和”__set()”来获取和赋值其属性,以及检查属性的”__isset()”和删除属性的方法”__unset()”。

上一节中,我们为每个属性做了设置和获取的方法,在PHP5中给我们提供了专门为属性设置值和获取值的方法,”__set()”和“__get()”这两个方法,这两个方法不是默认存在的, 而是我们手工添加到类里面去的,像构造方法(__construct())一样,类里面添加了才会存在,可以按下面的方式来添加这两个方法,当然也可以按个人的风格来添加:

​​<?php​​       


​​//__get()方法用来获取私有属性​​


​​function​​ ​​ __get(​​ ​​$property_name​​ ​​) ​​


​​if​​ ​​ (isset(​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​)) ​​


​​return​​ ​​ (​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​);​​


​​ ​​ ​​else​​ ​​ ​​


​​return​​ ​​ (NULL);​​


​​​​


​​​​





​​//__set()方法用来设置私有属性​​


​​function​​ ​​ __set(​​ ​​$property_name​​ ​​, ​​ ​​$value​​ ​​) ​​


​​$this​​ ​​->​​ ​​$property_name​​ ​​ = ​​ ​​$value​​ ​​;​​


​​​​


__get()方法:这个方法用来获取私有成员属性值的,有一个参数, 参数传入你要获取的成员属性的名称,返回获取的属性值, 这个方法不用我们手工的去调用, 是在直接获取私有属性的时候自动调用的。因为私有属性已经被封装上了,是不能直接获取值的(比如:”echo $p1->name” 这样直接获取是错误的),但是如果你在类里面加上了这个方法,在使用”echo $p1->name” 这样的语句直接获取值的时候就会自动调用__get($property_name)方法,将属性name传给参数$property_name,通过这 个方法的内部执行,返回我们传入的私有属性的值。

__set()方法:这个方法用来为私有成员属性设置值的, 有两个参数,第一个参数为你要为设置值的属性名,第二个参数是要给属性设置的值,没有返回值。这个方法同样不用我们手工去调用,是在直接设置私有属性值的 时候自动调用的,同样属性私有的已经被封装上了, 如果没有__set()这个方法,是不允许的, 比如:”$this->name=’zhangsan’,这样会出错,但是如果你在类里面加上了__set($property_name, $value)这个方法,在直接给私有属性赋值的时候,就会自动调用它,把属性比如name传给$property_name, 把要赋的值”zhangsan”传给$value,通过这个方法的执行,达到赋值的目的, 为了不传入非法的值, 还可以在这个方法给做一下判断。代码如下:

​​<?php​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性, 都是封装的私有成员​​


​​private​​ ​​ $name​​ ​​; ​​ ​​//人的名子​​


​​private​​ ​​ $sex​​ ​​; ​​ ​​//人的性别​​


​​private​​ ​​ $age​​ ​​; ​​ ​​//人的年龄​​





​​//__get()方法用来获取私有属性​​


​​function​​ ​​ __get(​​ ​​$property_name​​ ​​) ​​


​​echo​​ ​​ "在直接获取私有属性值的时候,自动调用了这个__get()方法<br />"​​ ​​;​​


​​if​​ ​​ (isset(​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​)) ​​


​​return​​ ​​ (​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​);​​


​​ ​​ ​​else​​ ​​ ​​


​​return​​ ​​ NULL;​​


​​​​


​​​​





​​//__set()方法用来设置私有属性​​


​​function​​ ​​ __set(​​ ​​$property_name​​ ​​, ​​ ​​$value​​ ​​) ​​


​​echo​​ ​​ "在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值<br />"​​ ​​;​​


​​$this​​ ​​->​​ ​​$property_name​​ ​​ = ​​ ​​$value​​ ​​;​​


​​​​


​​​​





​​$p1​​ ​​ = ​​ ​​new​​ ​​ Person();​​





​​// 直接为私有属性赋值的操作, 会自动调用__set()方法进行赋值​​


​​$p1​​ ​​->name = ​​ ​​"张三"​​ ​​;​​


​​$p1​​ ​​->sex = ​​ ​​"男"​​ ​​;​​


​​$p1​​ ​​->age = 20;​​





​​// 直接获取私有属性的值, 会自动调用__get()方法,返回成员属性的值​​


​​echo​​ ​​ "姓名:"​​ ​​ . ​​ ​​$p1​​ ​​->name . ​​ ​​"<br />"​​ ​​;​​


​​echo​​ ​​ "性别:"​​ ​​ . ​​ ​​$p1​​ ​​->sex . ​​ ​​"<br />"​​ ​​;​​


​​echo​​ ​​ "年龄:"​​ ​​ . ​​ ​​$p1​​ ​​->age . ​​ ​​"<br />"​​ ​​;​​


​​?>​​


程序执行结果:

在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值
在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值
在直接设置私有属性值的时候,自动调用了这个__set()方法为私有属性赋值
在直接获取私有属性值的时候,自动调用了这个__get()方法
姓名:张三
在直接获取私有属性值的时候,自动调用了这个__get()方法
性别:男
在直接获取私有属性值的时候,自动调用了这个__get()方法
年龄:20

以上代码如果不加上__get()和__set()方法,程序就会出错,因为不能在类的外部操作私有成员,而上面的代码是通过自动调用__get()和__set()方法来帮助我们直接存取封装的私有成员的。

__isset() 方法:在看这个方法之前我们看一下“isset()”函数的应用,isset()是测定变量是否设定用的函数,传入一个变量作为参数,如果传入的变量存在则传回true,否则传回false。那么如果在一个对象外面使用“isset()”这个函数去测定对象里面的成员是否被设定可不可以用它呢?分两种情况,如果对象里面成员是公有的,我们就可以使用这个函数来测定成员属性,如果是私有的成员属性,这个函数就不起作用了,原因就是因为私有的被封装了,在外部不可见。那么我们就不可以在对象的外部使用“isset()”函数来测定私有成员属性是否被设定了呢?可以,你只要在类里面加上一个“__isset()”方法就可以了,当在类外部使用”isset()”函数来测定对象里面的私有成员是否被设定时,就会自动调用类里面的“__isset()”方法了帮我们完成这样的操作,“__isset()”方法也可以做成私有的。你可以在类里面加上下面这样的代码就可以了:

​​private​​         ​​ function​​         ​​ __isset(​​        ​​$nm​​        ​​) ​​       


​​echo​​ ​​ "当在类外部使用isset()函数测定私有成员$nm时,自动调用<br />"​​ ​​;​​


​​return​​ ​​ isset(​​ ​​$this​​ ​​->​​ ​​$nm​​ ​​);​​


​​​​


__unset()方法:看这个方法之前呢,我们也先来看一下“unset()”这个函数,“unset()”这个函数的作用是删除指定的变量且传回true,参数为要删除的变量。那么如果在一个对象外部去删除对象内部的成员属性用“unset()”函数可不可以呢,也是分两种情况,如果一个对象里面的成员属性是公有的,就可以使用这个函数在对象外面删除对象的公有属性,如果对象的成员属性是私有的,我使用这个函数就没有权限去删除,但同样如果你在一个对象里面加上“__unset()”这个方法,就可以在对象的外部去删除对象的私有成员属性了。在对象里面加上了“__unset()”这个方法之后,在对象外部使用“unset()”函数删除对象内部的私有成员属性时,自动调用“__unset()”函数来帮我们删除对象内部的私有成员属性,这个方法也可以在类的内部定义成私有的。在对象里面加上下面的代码就可以了:

​​private​​         ​​ function​​         ​​ __unset(​​        ​​$nm​​        ​​) ​​       


​​echo​​ ​​ "当在类外部使用unset()函数来删除私有成员时自动调用的<br />"​​ ​​;​​


​​unset(​​ ​​$this​​ ​​->​​ ​​$nm​​ ​​);​​


​​​​


我们来看一个完整的实例:

​​<?php​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​private​​ ​​ $name​​ ​​; ​​ ​​//人的名子​​


​​private​​ ​​ $sex​​ ​​; ​​ ​​//人的性别​​


​​private​​ ​​ $age​​ ​​; ​​ ​​//人的年龄​​





​​// __get()方法用来获取私有属性​​


​​private​​ ​​ function​​ ​​ __get(​​ ​​$property_name​​ ​​) ​​


​​if​​ ​​ (isset(​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​)) ​​


​​return​​ ​​ (​​ ​​$this​​ ​​->​​ ​​$property_name​​ ​​);​​


​​ ​​ ​​else​​ ​​ ​​


​​return​​ ​​ NULL;​​


​​​​


​​​​





​​// __set()方法用来设置私有属性​​


​​private​​ ​​ function​​ ​​ __set(​​ ​​$property_name​​ ​​, ​​ ​​$value​​ ​​) ​​


​​$this​​ ​​->​​ ​​$property_name​​ ​​ = ​​ ​​$value​​ ​​;​​


​​​​





​​// __isset()方法​​


​​private​​ ​​ function​​ ​​ __isset(​​ ​​$nm​​ ​​) ​​


​​echo​​ ​​ "isset()函数测定私有成员时,自动调用<br />"​​ ​​;​​


​​return​​ ​​ isset(​​ ​​$this​​ ​​->​​ ​​$nm​​ ​​);​​


​​​​





​​//__unset()方法​​


​​private​​ ​​ function​​ ​​ __unset(​​ ​​$nm​​ ​​) ​​


​​echo​​ ​​ "当在类外部使用unset()函数来删除私有成员时自动调用的<br />"​​ ​​;​​


​​unset(​​ ​​$this​​ ​​->​​ ​​$nm​​ ​​);​​


​​​​


​​​​





​​$p1​​ ​​ = ​​ ​​new​​ ​​ Person();​​


​​$p1​​ ​​->name = ​​ ​​"this is a person name"​​ ​​;​​





​​// 在使用isset()函数测定私有成员时,自动调用__isset()方法帮我们完成,返回结果为true​​


​​echo​​ ​​ var_dump(isset(​​ ​​$p1​​ ​​->name)) . ​​ ​​"<br >"​​ ​​;​​


​​echo​​ ​​ $p1​​ ​​->name . ​​ ​​"<br />"​​ ​​;​​





​​// 在使用unset()函数删除私有成员时,自动调用__unset()方法帮我们完成,删除name私有属性​​


​​unset(​​ ​​$p1​​ ​​->name);​​





​​// 已经被删除了,所这行不会有输出​​


​​echo​​ ​​ $p1​​ ​​->name;​​


​​?>​​


输出结果为:

isset()函数测定私有成员时,自动调用
boolean true
this is a person name
当在类外部使用unset()函数来删除私有成员时自动调用的
isset()函数测定私有成员时,自动调用

__set()、__get()、__isset()、__unset() 这四个方法都是我们添加到对象里面的,在需要时自动调用的,来完成在对象外部对对象内部私有属性的操作。

 

最后补充说明:

不走__set(),__get()函数的!!!

2、在PHP5.3及以后,上述魔术方法(__get(),__set(),__isset(),__unset() 等)提倡是 public 类型的,并且不是 static 方法,否则会给出警告信息!

​​<?php​​       


​​class​​ ​​ A ​​


​​private​​ ​​ $name​​ ​​ = ​​ ​​qianyunlai​​ ​​;​​


​​public​​ ​​ $old​​ ​​ = ​​ ​​26​​ ​​;​​





​​private​​ ​​ function​​ ​​ __get(​​ ​​$name​​ ​​) ​​


​​echo​​ ​​ $name​​ ​​, ​​ ​​<br />​​ ​​;​​


​​return​​ ​​ $this​​ ​​->​​ ​​$name​​ ​​;​​


​​​​


​​​​





​​$a​​ ​​ = ​​ ​​new​​ ​​ A();​​


​​print_r(​​ ​​$a​​ ​​->name);​​


​​?>​​


输出:

( ! ) Warning: The magic method __get() must have public visibility and cannot be static in D:\\PHP\\xampp\\htdocs\\discuz\\discuzx3.0\\123.php on line 6

类的继承

下面是“人”类的抽象

​​// 定义一个“人”类做为父类​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​var​​ ​​ $name​​ ​​; ​​ ​​//人的名子​​


​​var​​ ​​ $sex​​ ​​; ​​ ​​//人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​//人的年龄​​





​​// 定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​) ​​


​​$this​​ ​​->name = ​​ ​​$name​​ ​​;​​


​​$this​​ ​​->sex = ​​ ​​$sex​​ ​​;​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​​​





​​// 这个人可以说话的方法, 说出自己的属性​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​


​​​​


​​​​


下面我们做一个“学生类”,如果不是用继承如下:

​​class​​         ​​ Student ​​       


​​// 下面是人的成员属性​​


​​var​​ ​​ $name​​ ​​; ​​ ​​// 人的名字​​


​​var​​ ​​ $sex​​ ​​; ​​ ​​// 人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​// 人的年龄​​


​​var​​ ​​ $school​​ ​​; ​​ ​​// 学生所在学校的属性​​





​​// 定义一个构造方法参数为属性姓名$name、性别$sex 和年龄$age 进行赋值​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​ = ​​ ​​""​​ ​​, ​​ ​​$sex​​ ​​ = ​​ ​​""​​ ​​, ​​ ​​$age​​ ​​ = ​​ ​​""​​ ​​, ​​ ​​$school​​ ​​ = ​​ ​​""​​ ​​) ​​


​​$this​​ ​​->name = ​​ ​​$name​​ ​​;​​


​​$this​​ ​​->sex = ​​ ​​$sex​​ ​​;​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​$this​​ ​​->school = ​​ ​​$school​​ ​​;​​


​​​​





​​// 这个人可以说话的方法, 说出自己的属性​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名字叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age . ​​ ​​"<br />"​​ ​​;​​


​​​​





​​// 这个学生学习的方法​​


​​function​​ ​​ study() ​​


​​echo​​ ​​ "我的名字叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 我正在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"学习<br />"​​ ​​;​​


​​​​


​​​​


定义一个子类“学生类“使用”extends”关键字来继承”人”类:

​​class​​         ​​ Student ​​        ​​extends​​         ​​ Person ​​       


​​var​​ ​​ $school​​ ​​; ​​ ​​// 学生所在学校的属性​​





​​// 这个学生学习的方法​​


​​function​​ ​​ study() ​​


​​echo​​ ​​ "我的名字叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 我正在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"学习<br />"​​ ​​;​​


​​​​


​​​​


通过上面“Student“类的定义, Student类通过使用”extends”这个关键字把Person 类里的所有成员属性和成员方法都继承过来了,并扩展了一个所在学校成员属性”school”,和一个学习方法“study()”。现在子类”Student”里面和使用这个类实例出来的对象都具有如下的属性和方法:

学生类”Student”里面的成员属性有:

  • 姓名:name;
  • 年龄:age;
  • 性别:sex;
  • 学校:school;
  • 学生类”Student”里面的成员方法有:
  • 说话方法:say();
  • 学习方法:study();

通过上面类继承的使用简化了对象、类的创建工作量,增加了代码的可重性。但是从上面这一个例子上中“可重用性”以及其它的继承性所带来的影响,我们看的还不是特别的明显,你扩展的去想一下,人有无数个岗位,比如上面的学生还有老师、工程师、医生、工人等,很多很多,如果

每个类都定义“人”都共同具有的属性和方法,想一想会有很大的工作量,这些属性和方法都可以从“Person”人类里面继承过来。

 

虽然说在PHP里面不能定义同名的方法, 但是在父子关系的两个类中,我们可以在子类中定义和父类同名的方法,这样就把父类中继承过来的方法覆盖掉了。

​​<?​​       


​​// 定义一个"人"类做为父类​​


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​var​​ ​​ $name​​ ​​; ​​ ​​// 人的名子​​


​​var​​ ​​ $sex​​ ​​; ​​ ​​// 人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​// 人的年龄​​





​​// 定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​) ​​


​​$this​​ ​​->name = ​​ ​​$name​​ ​​;​​


​​$this​​ ​​->sex = ​​ ​​$sex​​ ​​;​​


​​$this​​ ​​->age = ​​ ​​$age​​ ​​;​​


​​​​





​​// 这个人可以说话的方法, 说出自己的属性​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age;​​


​​​​


​​​​





​​class​​ ​​ Student ​​ ​​extends​​ ​​ Person ​​


​​var​​ ​​ $school​​ ​​; ​​ ​​// 学生所在学校的属性​​





​​// 这个学生学习的方法​​


​​function​​ ​​ study() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 我正在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​" 学习"​​ ​​;​​


​​​​





​​// 这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法​​


​​function​​ ​​ say() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 性别:"​​ ​​ . ​​ ​​$this​​ ​​->sex . ​​ ​​" 我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age . ​​ ​​" 我在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"上学"​​ ​​;​​


​​​​


​​​​


​​?>​​


上面的例子, 我们就在“Student”子类里覆盖了继承父类里面的”say()”的方法,通过覆盖我们就实现了对“方法”扩展。但是,像这样 做虽然解决了我们上面说的问题,但是在实际开发中,一个方法不可能就一条代码或是几条代码,比如说“Person”类里面的“say()”方法有里面有 100条代码,如果我们想对这个方法覆盖保留原有的功能外加上一点点功能,就要把原有的100条代码重写一次, 再加上扩展的几条代码,这还算是好的,而有的情况,父类中的方法是看不见原代码的,这个时候你怎么去重写原有的代码呢?我们也有解决的办法,就是在子类这 个方法中可以调用到父类中被覆盖的方法, 也就是把被覆盖的方法原有的功能拿过来再加上自己的一点功能,可以通过两种方法实现在子类的方法中调用父类被覆盖的方法:

一种是使用父类的“类名::“来调用父类中被覆盖的方法;

一种是使用“parent::”的方试来调用父类中被覆盖的方法;

​​class​​         ​​ Student ​​        ​​extends​​         ​​ Person ​​       


​​var​​ ​​ $school​​ ​​; ​​ ​​// 学生所在学校的属性​​





​​// 这个学生学习的方法​​


​​function​​ ​​ study() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 我正在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"学习"​​ ​​;​​


​​​​





​​// 这个学性可以说话的方法, 说出自己所有的属性,覆盖了父类的同名方法​​


​​function​​ ​​ say() ​​





​​// 使用父类的"类名::"来调用父类中被覆盖的方法;​​


​​// Person::say();​​





​​// 或者使用"parent::"的方试来调用父类中被覆盖的方法;​​


​​parent::say();​​





​​// 加上一点自己的功能​​


​​echo​​ ​​ "我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age . ​​ ​​" 我在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"上学"​​ ​​;​​


​​​​


​​​​


现在用两种方式都可以访问到父类中被覆盖的方法,我们选那种方式最好呢?用户可能会发现自己写的代码访问了父类的变量和函数。如果子类非常精炼或者父类非 常专业化的时候尤其是这样。 不要用代码中父类文字上的名字,应该用特殊的名字 parent,它指的就是子类在 extends 声明中所指的父类的名字。这样做可以避免在多个地方使用父类的名字。如果继承树在实现的过程中要修改,只要简单地修改类中 extends 声明的部分。
同样,构造方法在子类中如果没有声明的话,也可以使用父类中的构造方法,如果子类中重新定义了一个构造方法也会覆盖掉父类中的构造方法,如果想使用新的构造方法为所有属性赋值也可以用同样的方式。

​​class​​         ​​ Student ​​        ​​extends​​         ​​ Person ​​       


​​var​​ ​​ $school​​ ​​; ​​ ​​// 学生所在学校的属性​​





​​function​​ ​​ __construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​, ​​ ​​$school​​ ​​) ​​


​​// 使用父类中的方法为原有的属性赋值​​


​​parent::__construct(​​ ​​$name​​ ​​, ​​ ​​$sex​​ ​​, ​​ ​​$age​​ ​​);​​


​​$this​​ ​​->school = ​​ ​​$school​​ ​​;​​


​​​​





​​// 这个学生学习的方法​​


​​function​​ ​​ study() ​​


​​echo​​ ​​ "我的名子叫:"​​ ​​ . ​​ ​​$this​​ ​​->name . ​​ ​​" 我正在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​" 学习"​​ ​​;​​


​​​​





​​// 这个人可以说话的方法, 说出自己的属性​​


​​function​​ ​​ say() ​​


​​parent::say();​​





​​// 加上一点自己的功能​​


​​echo​​ ​​ "我的年龄是:"​​ ​​ . ​​ ​​$this​​ ​​->age . ​​ ​​" 我在"​​ ​​ . ​​ ​​$this​​ ​​->school . ​​ ​​"上学"​​ ​​;​​


​​​​


​​​​



所以“__toString()”方法一定要有个返回值(return 语句)。

​​<?php​​       


​​// Declare a simple class​​


​​class​​ ​​ TestClass ​​


​​public​​ ​​ $foo​​ ​​;​​





​​public​​ ​​ function​​ ​​ __construct(​​ ​​$foo​​ ​​) ​​


​​$this​​ ​​->foo = ​​ ​​$foo​​ ​​;​​


​​​​





​​// 定义一个__toString方法,返加一个成员属性$foo​​


​​public​​ ​​ function​​ ​​ __toString() ​​


​​return​​ ​​ $this​​ ​​->foo;​​


​​​​


​​​​





​​$class​​ ​​ = ​​ ​​new​​ ​​ TestClass(​​ ​​Hello​​ ​​);​​





​​// 直接输出对象​​


​​echo​​ ​​ $class​​ ​​;​​


​​?>​​


上例输出:Hello



$this指向复本,而$that指向原本;

​​<?​​       


​​class​​ ​​ Person ​​


​​// 下面是人的成员属性​​


​​var​​ ​​ $name​​ ​​; ​​ ​​// 人的名子​​


​​var​​ ​​ $sex​​ ​​; ​​ ​​// 人的性别​​


​​var​​ ​​ $age​​ ​​; ​​ ​​// 人的年龄​​





​​// 定义一个构造方法参数为属性姓名$name、性别$sex和年龄$age进行赋值​​


​​function​​ ​​ __construct(​​ ​​$name​​ ​​ = ​​ ​​""​​ ​​, ​​ ​​$sex​​ ​​ = ​​ ​​""​​ ​​, ​​ ​​$age​​ ​​ = ​​ ​

以上是关于php面向对象(OOP)编程的主要内容,如果未能解决你的问题,请参考以下文章

面向对象基础

PHP面向对象OOP编程

php面向对象编程(oop)基础知识

PHP的面向对象编程思想

php面向对象

PHP面向对象(OOP)编程入门教程3.什么是面向对象编程呢?