构造函数详解跟析构函数与对象的回收机制

在类中, 有一个构造函数,

就是用来初始化对象用的.

利用构造函数,你有机会操作对象,

并改变他的值

相关:

 

 

PHP异常处理

异常处理(又称为错误处理)功能提供了处理程序运行时出现的错误或异常情况的方法。

异常处理通常是防止未知错误产生所采取的处理措施。异常处理的好处是你不用再绞尽脑汁去考虑各种错误,这为处理某一类错误提供了一个很有效的方法,使编程效率大大提高。当异常被触发时,通常会发生:

当前代码状态被保存

代码执行被切换到预定义的异常处理器函数

根据情况,处理器也许会从保存的代码状态重新开始执行代码,终止脚本执行,或从代码中另外的位置继续执行脚本

PHP 5 提供了一种新的面向对象的错误处理方法。可以使用检测(try)、抛出(throw)和捕获(catch)异常。即使用try检测有没有抛出(throw)异常,若有异常抛出(throw),使用catch捕获异常。

一个 try 至少要有一个与之对应的 catch。定义多个 catch 可以捕获不同的对象。PHP 会按这些 catch 被定义的顺序执行,直到完成最后一个为止。而在这些 catch 内,又可以抛出新的异常。

异常

PHP类的自动加载

__autoload用法


__autoload()是实现了一个自动加载的功能;

当你去实例化一个未声明的类时,会自动调用__autoload()方法;

如果new时,该类已经声明,__autoload()方法不执行;

__autoload()方法被系统自动调用时,收到一个一个参数,值为类名!
HumanModel.php

  1. <?php
  2. // HumanModel类
  3. class HumanModel {
  4.     public function t() {
  5.         echo ‘人类’;
  6.     }
  7. }

01.php

  1. <?php
  2. require(‘./HumanModel.php’);
  3. $lisi = new HumanModel();
  4. $lisi->t();
  5. //人类
  6. /*
  7. 如上,没有require引入进来时,报错
  8. 手动require进来
  9. 如果网站比较大,model类比较多,如何自动加载进来?
  10. HumanModel
  11. UserModel
  12. GoodsModel
  13. CatModel
  14. OrderModel
  15. 1:这么多的model,我用谁,就得include/require谁.
  16. 2:而且不知道,之前是否已经include/require进来某个类
  17. (这个用once可以解决,但once的效率很低)
  18. 这时 我们可以用自动加载!
  19. */
  20. echo ‘<br />’;
  21. function __autoload($c) {
  22.     echo ‘~~~~’,$c,’~~~~~’;
  23. }
  24. $ming = new Pig();
  25. //~~~~Pig~~~~~
  26. //( ! ) Fatal error: Class ‘Pig’ not found in D:\wamp\www\1115\05.php on line 45
  27. /*
  28. 如果调用某个不存在的类,
  29. 在报错之前,
  30. 我们还有一次介入机会 __autoload函数
  31. 系统会调用__autoload()函数,
  32. 并把”类名”自动传给__autoload函数
  33. 我们自然可以在__autoload里 加载需要的类!
  34. 见下页
  35. */

02.php

  1. <?php
  2. /***
  3. __autoload用法
  4. __autoload()是实现了一个自动加载的功能;
  5. 当你去实例化一个为声明的类时,会自动调用__autoload()方法;
  6. 如果new时,该类已经声明,__autoload()方法不执行;
  7. __autoload()方法被系统自动调用时,收到一个一个参数,值为类名!
  8. ***/
  9. function __autoload($c) {
  10.     echo ‘我先自动加载’;
  11.     echo ‘./’ . $c . ‘.php’;
  12.     echo ‘<br />’;
  13.     require(‘./’ . $c . ‘.php’);
  14. }
  15. $lisi = new HumanModel();
  16. $lisi->t();
  17. /*我先自动加载./HumanModel.php
  18. 人类*/
  1. <?php
  2. /***
  3. ====笔记部分====
  4. ***/
  5. function test() {
  6.     // 函数内可以写任何合法的PHP代码,包含再声明一个函数/类
  7.     echo ‘来’;
  8.     class Bird {
  9.         public static function sing() {
  10.             echo ‘百灵鸟儿放声唱!’;
  11.         }
  12.     }
  13.     echo ‘去’;
  14. }
  15. // Bird::sing();   // Class ‘Bird’ not found
  16. test();
  17. //函数要调用才能执行! 
  18. Bird::sing();
  19. //来去百灵鸟儿放声唱!
  1. <?php
  2. /***
  3. ====自定义函数注册成为”自动加载函数”====
  4. ——–>自动加载只能用__autoload函数吗?<——–
  5.             答:不是的,其实也可以指定一个函数
  6. 比如:我们就用zidongjiazai()函数
  7. 注意:
  8. 要通知系统,让系统知道–我自己写了一个自动加载方法,用这个!
  9. 怎么通知: 用系统函数 spl_auto_register
  10. ***/
  11. // 下面这句话,是把zidongjiazai函数注册成为”自动加载函数”;
  12. spl_autoload_register(‘zidongjiazai’);
  13. function zidongjiazai($c) {
  14.     echo ‘我引入了./’ .  $c . ‘.php’,'<br />’;
  15.     require(‘./’ .  $c . ‘.php’);
  16. }
  17. $HumanModel = new HumanModel();
  18. $HumanModel->t();
  19. /**
  20. __autoload 是一个函数
  21. 我能自己注册一个自动加载函数
  22. 能否注册类的一个静态方法 当 自动加载函数?
  23. TP里这么做的,自己解决 🙂
  24. **/

PHP接口的概念跟接口语言与接口的应用场景

类: 是某一类事物的抽象,是某类对象的蓝图.


举例:

比如: 女娲造人时,脑子中关于人的形象 就是人类 class Human

如果,女娲决定造人—->形象又没最终定稿时,—->她脑子有哪些支离破碎的形象呢?

她可能会这么思考:

动物: 会吃饭

猴子: 会奔跑

猴子: 会哭

自己: 会思考

小鸟:会 飞

定义如下功能:

eat()

run();

cry();

think();

interface:接口

重点说明:

类如果是一种事物/【动物】的抽象

那么 接口,则是事物/动物———-【功能】的抽象,

功能各拆成小块——>自由组合成新的物种

接口的具体语法:


接口说明

1:以人类为, class Human 是人的草图

接口 是零件—>可以用多种零件组合出一种新物种来.

2: 如上,[接口本身]即是———抽象的,

[内部声明的方法] 默认也是———抽象的,不用加 abstract

3: 一个类可以一次性实现多个接口.

语法用【 implements 实现 】(把我这几个功能实现了)

class ClassName implements interface1,interface2,interface3 {

}

然后再把接口的功能给实现.

4 接口也可以继承, 用extends

5:接口是一堆方法的说明,不能加属性

6:接口就是供组装成类用的,封闭起来没有意义,因此方法只能是public

面向对象的一个观点:


做的越多,越容易犯错

抽象类{就定义类模板}—————>具体子类实现{语言展现china,japan,english}

接口:

抽象的数据库类

举例:

———–>创业做网站:

到底用什么数据库? mysql, oracle,sqlserver,postgresql?

这样:先开发网站,运行再说.

先弄个mysql开发着,正式上线了再换数据库也不迟

———–>引来问题:

换数据库,会不会以前的代码又得重写?

答:不必,用抽象类

———–>开发者,开发时,就以db抽象类来开发.

下面这个代码有误

因为子类实现时, connect和抽象类的connect参数不一致

正确的严格的抽象类调用!

接口 就更加抽象了

比如一个社交网站,

关于用户的处理是核心应用.

登陆

退出

写信

看信

招呼

更换心情

吃饭

骂人

捣乱

示爱

撩骚

这么多的方法,都是用户的方法,

自然可以写一个user类,全包装起来

但是,分析用户一次性使不了这么方法

用户信息类:{登陆,写信,看信,招呼,更换心情,退出}

用户娱乐类:{登陆,骂人,捣乱,示爱,撩骚,退出}

开发网站前,分析出来这么多方法,

但是,不能都装在一个类里,

分成了2个类,甚至更多.

作用应用逻辑的开发,这么多的类,这么多的方法,都晕了.

接口的规范作用

推荐阅读:

PHP抽象类:无法实例化

PHP延迟绑定练习

普通常量_魔术常量_后期绑定/延迟绑定

PHP抽象类:无法实例化

一、抽象类: 无法实例化


类前加 abstract, 此类就成为抽象类,无法实例化.

举例:

春秋战国时期,燕零七 飞行器专家,能工巧匠.他写了一份图纸—【飞行器制造术】

飞行器秘制图谱:

  • 要有一个有力的发动机,喷气式.
  • 要有一个平衡舵,掌握平衡

他的孙子问: 发动机怎么造呢?

燕零七眼望夕阳: 我是造不出来,但我相信后代有人造出来

总结:

  类前加 abstract 是抽象类

方法前加 abstract 是抽象方法

抽象类 不能 实例化

抽象方法 不能有 方法体

有抽象方法,则此类必是 抽象类

抽象类,内未必有抽象方法

但是 — 即便全是具体方法,但类是抽象的,

也不能实例化.

二、抽象类的意义


请看如下场景:

Facebook 多国语言欢迎页面

user登陆,有一个 c 字段,是其国家

当各国人登陆时,看到各国语言的欢迎界面

我们可以用面向过程的来做

反思: 当facebook进入泰国市场时,

增加 else if ,扩展性很差

=====用面向对象来做======

让美国小组/中国开发组/斯蜜达开发组 来开发Welcome类

争执不下: echo 到底该中? 日? 韩?

说: 干脆在wel()方法里,判断一下? 没意义啊

 

普通常量_魔术常量_后期绑定/延迟绑定

普通常量


define(‘常量名’,常量值);

以前说过: define定义的常量 ,全局有效.无论是页面内,函数内,类内,都可以访问.

能否定义 专门在类内发挥作用的常量?

专门在类内发挥作用 说明:

  1. 作用域在类内,类似于静态属性
  2. 又是常量,则不可改.

【其实就是”不可改变的静态属性”】

类常量 在类内用 const 声明即可,前面不用加修饰符,而且权限是public的,即外部也可以访问

魔术常量:


  1. 无法手动修改他的值,所以叫常量
  2. 但是值又是随环境变动的,所以叫魔术

魔术常量:

  __FILE__ 返回当前文件的路径.

在框架开发或者是网站初始化脚本中,用来计算网站的根目录

__LINE__ 返回当前的行号

在框架中,可以用来在debug时,记录错误信息

__CLASS__ 返回当前的类名

__METHOD__ 返回当前的方法名

后期绑定/延迟绑定


 

PHP面向对象之重写与重载

重写/覆盖:override

  指:子类重写了父类的同名方法

重载: overload

  重载是指:存在多个同名方法,但参数类型/个数不同.

传(几个)不同的参数,调用不同的方法

但是在PHP中,不允许存在多个同名方法.

因此,不能够完成java,c++中的这种重载

但是,PHP的灵活,能达到类似的效果

 

__call__callStatic方法及如何ThinkPHP项目中应用

__call 当要调用的方法不存在或权限不足时,会自动调用__call 方法。

__callStatic 当调用的静态方法不存在或权限不足时,会自动调用__callStatic方法。

__call 在thinkPHP中的应用

 

php之魔术方法在ThinkPHP框架的应用

 

魔术方法__set__get__unset__isset讲解

魔术方法:

是指某些情况下,会自动调用的方法,称为魔术方法

PHP面向对象中,提供了这几个魔术方法,

他们的特点 都是以双下划线__开头的

  __construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(), __sleep(), __wakeup(), __toString(), __invoke(), __set_state() 和 __clone()

__clone() :克隆方法,当对象被克隆时,将会自动调用

__get( $property ) 当调用一个未定义的属性时访问此方法

__set( $property, $value ) 给一个未定义的属性赋值时调用

__isset( $property ) 当在一个未定义的属性上调用isset()函数时调用此方法

__unset( $property ) 当在一个未定义的属性上调用unset()函数时调用此方法

PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep, __wakeup, __toStr

针对php面向对象中的单例模式总结

  

单例模式思路总结:

  1. 保护或私有构造函数,防止外部实例化。
  2. 类内部放一个公共的静态方法,负责实例化。
  3. 类有一个静态属性,储存对象。
  4. 当静态属性已经有一个对象,直接return回去。

先看场景:

多人协同开发, 都要调用mysql类的实例! 如果用svn知道,好多人一起开发,再提交各自的文件.

  A:

$mysql = new mysql();

$mysql->query….

测试通过

B:

$db = new mysql();

测试通过

两人的代码要合到一块,如下

  $mysql = new mysql();

$mysql->query….

$db = new mysql();

两个mysql类的实例,

而且,每new一下,还要连接一次数据库.

显然,一个页面呢,有一个mysql类的实例就够了.

如果限制,让多人开发,无论你怎么操作,只能得到一个对象呢?

  1:开会时,经理说:有一个$db变量,是系统自动初始化的,就是mysql类的实例.

大家都用他.谁敢new mysql(),开除.

2:这是行政手段,不能阻止技术上的new mysql()行为.

我们可以从技术上,用单例模式来解决

注:单例常用也常考,请认真练习

  第一步:一个普通的类

这个普通类,可以new 来实例化

这显然不是单例

  第二步:看来new是罪恶之源,干脆不让new了

我们把构造方法 保护/私有

外部不能new了

–但引出一个问题,不能new,那得不到对象,这不是单例,这是0例模式

  第三部,通过内部的static方法,来调用

  第四步,通过内部的static方法实例化,

并且,把实例保存在类内部的静态属性上

  看问题


无标题

05.php

 

PHP中的魔术方法总结 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep, __wakeup, __toStr

1、__get、__set
这两个方法是为在类和他们的父类中没有声明的属性而设计的
__get( $property ) 当调用一个未定义的属性时访问此方法
__set( $property, $value ) 给一个未定义的属性赋值时调用
这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)

2、__isset、__unset
__isset( $property ) 当在一个未定义的属性上调用isset()函数时调用此方法
__unset( $property ) 当在一个未定义的属性上调用unset()函数时调用此方法
与__get方法和__set方法相同,这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)
3、__call
__call( $method, $arg_array ) 当调用一个未定义的方法是调用此访求
这里的未定义的方法包括没有权限访问的方法

4、__autoload
__autoload 函数,它会在试图使用尚未被定义的类时自动调用。通过调用此函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类。
注意: 在 __autoload 函数中抛出的异常不能被 catch 语句块捕获并导致致命错误。

5、__construct、__destruct
__construct 构造方法,当一个对象创建时调用此方法,使用此方法的好处是:可以使构造方法有一个独一无二的名称,无论它所在的类的名称是什么.这样你在改变类的名称时,就不需要改变构造方法的名称
__destruct 析构方法,PHP将在对象被销毁前(即从内存中清除前)调用这个方法
默认情况下,PHP仅仅释放对象属性所占用的内存并销毁对象相关的资源.
析构函数允许你在使用一个对象之后执行任意代码来清除内存.
当PHP决定你的脚本不再与对象相关时,析构函数将被调用.
在一个函数的命名空间内,这会发生在函数return的时候.
对于全局变量,这发生于脚本结束的时候.如果你想明确地销毁一个对象,你可以给指向该对象的变量分配任何其它值.通常将变量赋值勤为NULL或者调用unset.

6、__clone
PHP5中的对象赋值是使用的引用赋值,如果想复制一个对象则需要使用clone方法,在调用此方法是对象会自动调用__clone魔术方法
如果在对象复制需要执行某些初始化操作,可以在__clone方法实现

7、__toString
__toString方法在将一个对象转化成字符串时自动调用,比如使用echo打印对象时
如果类没有实现此方法,则无法通过echo打印对象,否则会显示:Catchable fatal error: Object of class test could not be converted to string in
此方法必须返回一个字符串

在PHP 5.2.0之前,__toString方法只有结合使用echo() 或 print()时 才能生效。PHP 5.2.0之后,则可以在任何字符串环境生效(例如通过printf(),使用%s修饰符),但 不能用于非字符串环境(如使用%d修饰符)。从PHP 5.2.0,如果将一个未定义__toString方法的对象 转换为字符串,会报出一个E_RECOVERABLE_ERROR错误。

8、__sleep、__wakeup
__sleep 串行化的时候用
__wakeup 反串行化的时候调用
serialize() 检查类中是否有魔术名称 __sleep 的函数。如果这样,该函数将在任何序列化之前运行。它可以清除对象并应该返回一个包含有该对象中应被序列化的所有变量名的数组。
使用 __sleep 的目的是关闭对象可能具有的任何数据库连接,提交等待中的数据或进行类似的清除任务。此外,如果有非常大的对象而并不需要完全储存下来时此函数也很有用。
相反地,unserialize() 检查具有魔术名称 __wakeup 的函数的存在。如果存在,此函数可以重建对象可能具有的任何资源。
使用 __wakeup 的目的是重建在序列化中可能丢失的任何数据库连接以及处理其它重新初始化的任务。

9、__set_state
当调用var_export()时,这个静态 方法会被调用(自PHP 5.1.0起有效)。
本方法的唯一参数是一个数组,其中包含按array(’property’ => value, …)格式排列的类属性。

10、__invoke
当尝试以调用函数的方式调用一个对象时,__invoke 方法会被自动调用。
PHP5.3.0以上版本有效
11、__callStatic
它的工作方式类似于 __call() 魔术方法,__callStatic() 是为了处理静态方法调用,
PHP5.3.0以上版本有效
PHP 确实加强了对 __callStatic() 方法的定义;它必须是公共的,并且必须被声明为静态的。同样,__call() 魔术方法必须被定义为公共的,所有其他魔术方法都必须如此。

总结self, parent的用法

  •   self: 本类(是类 不是对象)
  •   parent: 父类

php中static静态变量的使用方法详解

  • 类->访问->静态方法 可以
  • 类->动态方法 方法内没有this的情况下,但严重不支持.逻辑上解释不通.【无对象】
  • 对象–>访问动态方法 可以
  • 对象–>静态方法 可以
  • 类的静态属性与普通属性

精讲多态 PHP与java对比讲解

java多态