1.Abstract class(抽象类)

  • 抽象类是指在 class 前加了 abstract 关键字且存在抽象方法(在类方法 function 关键字前加了 abstract 关键字)的类。
  • 抽象类不能被直接实例化。抽象类中只定义(或部分实现)子类需要的方法。子类可以通过继承抽象类并通过实现抽象类中的所有抽象方法,使抽象类具体化。
  • 如果子类需要实例化,前提是它实现了抽象类中的所有抽象方法。如果子类没有全部实现抽象类中的所有抽象方法,那么该子类也是一个抽象类,必须在 class 前面加上 abstract 关键字,并且不能被实例化。

举例说明

  1. abstract Class AbstractClass
  2. {
  3. abstract public function A();//这里不要携带body
  4. abstract protected function B();//这里不要携带body
  5. public function C(){}//这里要携带body
  6. }
  7.  
  8. class ClassA extends AbstractClass
  9. {
  10. public function A(){
  11. echo "Hello,I'm A <br/>";
  12. }
  13. protected function B(){
  14. echo "Hello,I'm B <br/>";
  15. }
  16. public function E()
  17. {
  18. echo "Hello,I'm E <br/>";
  19. }
  20. }

注意要点:

  • 如果 AbstractClass实现了抽象方法 B() ,那么 ClassA  中 B() 方法的访问控制不能比 AbstractClass 中 B() 的访问控制更严格,也就是说:

    1)、如果AbstractClass中B为Public,则ClassA中B只能为public

    2)、如果AbstractClass中B为Protected,则ClassA中B只能为public或Protected

    3)、注意,AbstractClass的抽象方法不能为Private

  • 抽象方法与普通的方法不一样,它只是子类中普通方法的一个占位符(只是占个地主不启作用),没有任何代码,也没有"{}"包含,而是以";"结束的.

2.Interface(接口)

  • 接口用关键字 interface 来声明。抽象类提供了具体实现的标准,而接口则是纯粹的模版。接口只定义功能,而不包含实现的内容。
  • interface 是完全抽象的,只能声明方法,而且只能声明 public 的方法,不能声明 private 及 protected 的方法,不能定义方法体,也不能声明实例变量 。然而, interface 却可以声明常量变量 。但将常量变量放在 interface 中违背了其作为接口的作用而存在的宗旨,也混淆了 interface 与类的不同价值。如果的确需要,可以将其放在相应的 abstract class 或 Class 中。
  • 任何实现接口的类都要实现接口中所定义的所有方法,否则该类必须声明为 abstract 。
  • 一个类可以在声明中使用 implements 关键字来实现某个接口。这么做之后,实现接口的具体过程和继承一个仅包含抽象方法的抽象类是一样的。一个类可以同时继承一个父类和实现任意多个接口。 extends 子句应该在 implements 子句之前。 PHP 只支持继承自一个父类,因此 extends 关键字后只能跟一个类名。
  • 接口不可以实现另一个接口,但可以继承多个

实例:

  1. interface A
  2. {
  3. public function fA();
  4. Public function fB();
  5. }
  6. interface B
  7. {
  8. public function fC();
  9. Public function fD();
  10. }
  11. interface C extends A,B
  12. {
  13.  
  14. }
  15. class M implements C
  16. {
  17. public function fA(){
  18.  
  19. }
  20. public function fB(){
  21.  
  22. }
  23. public function fC(){
  24.  
  25. }
  26. public function fD(){
  27.  
  28. }
  29. }

3.Abstract Class与Interface的异同

相同点

  • 两者都是抽象类,都不能实例化。
  • interface 实现类及 abstract class 的子类都必须要实现已经声明的抽象方法。

不同点

  • interface 需要实现,要用 implements ,而 abstract class 需要继承,要用 extends 。
  • 一个类可以实现多个 interface ,但一个类只能继承一个 abstract class 。
  • interface 强调特定功能的实现,而 abstract class 强调所属关系。
  • 尽管 interface 实现类及 abstract class 的子类都必须要实现相应的抽象方法,但实现的形式不同。 interface 中的每一个方法都是抽象方法,都只是声明的 (declaration, 没有方法体 ) ,实现类必须要实现。

     而 abstract class 的子类可以有选择地实现。这个选择有两点含义:

    a) abstract class 中并非所有的方法都是抽象的,只有那些冠有 abstract 的方法才是抽象的,子类必须实现。那些没有 abstract 的方法,在 abstract class 中必须定义方法体;

    b) abstract class 的子类在继承它时,对非抽象方法既可以直接继承,也可以覆盖;而对抽象方法,可以选择实现,也可以留给其子类来实现,但此类必须也声明为抽象类。既是抽象类,当然也不能实例化。

  • abstract class 是 interface 与 class 的中介。 abstract class 在 interface 及 class 中起到了承上启下的作用。一方面, abstract class 是抽象的,可以声明抽象方法,以规范子类必须实现的功能;另一方面,它又可以定义缺省的方法体,供子类直接使用或覆盖。另外,它还可以定义自己的实例变量,以供子类通过继承来使用。
  • 接口中的抽象方法前不用也不能加 abstract 关键字,默认隐式就是抽象方法,也不能加 final 关键字来防止抽象方法的继承。而抽象类中抽象方法前则必须加上 abstract 表示显示声明为抽象方法。
  • 接口中的抽象方法默认是 public 的,也只能是 public 的,不能用 private , protected 修饰符修饰。而抽象类中的抽象方法则可以用 public , protected 来修饰,但不能用 private 。

引用与推荐阅读>> PHP中的 抽象类(abstract class)和 接口(interface)

4.Abstract关键字

abstract关键字用于定义抽象方法与抽象类。

抽象方法指:没有方法体的方法,具体就是在方法声明的时候没有{},而是直接分号结束。一般用abstract定义的方法被称为抽象方法。在interface中所有的方法都是抽象方法。

定义抽象方法形如:

  1. abstract function func_name();

抽象类指:只要一个类里面有一个方法是抽象方法,那么这个类就定义为抽象类。抽象类同样用abstract关键字来定义。关于抽象类的概念,更多了解可以看上面的解释。

5.Final关键字

final关键字,如果父类中的方法被声明为final,则子类无法覆盖该方法,如果一个类被声明为final,则不能被继承。

6.Static关键字

static关键字用来定义静态方法和属性,也可以用来定义静态变量以及后期静态绑定。

  • 静态方法和属性:

  声明类属性或方法为静态,就可以不实例化类而直接访问。静态属性不能通过一个类已实例化的对象来访问(但静态方法可以)。在没有指定访问控制(public,private,protected)的时候,属性和方法默认public。

  静态方法不需要通过对象即可调用,所以伪变量$this在静态方法中不可用。

  静态属性不可以由对象通过 -> 操作符来访问。

  用静态方式调用一个非静态方法会导致一个 E_STRICT 级别的错误。

  就像其它所有的 PHP 静态变量一样,静态属性只能被初始化为文字或常量,不能使用表达式。所以可以把静态属性初始化为整数或数组,但不能初始化为另一个变量或函数返回值,也不能指向一个对象。

  自 PHP 5.3.0 起,可以用一个变量来动态调用类。但该变量的值不能为关键字 self,parent 或 static。

代码说明:

\Libs\Foo.php

  1. <?php
  2. namespace Libs;
  3.  
  4. class Foo
  5. {
  6. public static $my_static = 'foo';
  7. public function staticFunc()
  8. {
  9. return self::$my_static;
  10. }
  11. }

\Libs\SonFoo.php

  1. <?php
  2. namespace Libs;
  3.  
  4. class SonFoo extends Foo
  5. {
  6. public function sonStatic()
  7. {
  8. return parent::$my_static;
  9. }
  10. }

\Libs\UseFoo.php

  1. <?php
  2. namespace Libs;
  3.  
  4. class UseFoo
  5. {
  6. static public function index()
  7. {
  8. $eof = "<br/>";
  9.  
  10. echo Foo::$my_static,1,$eof; //静态属性的调用方式
  11.  
  12. $foo = new Foo();
  13. echo $foo->staticFunc(),2,$eof;
  14.  
  15. echo $foo->my_static,3,$eof;//报错,因为不能使用->方式调用静态属性
  16.  
  17. echo $foo::$my_static,4,$eof;
  18.  
  19. $newfoo = 'Libs\Foo';
  20. echo $newfoo::$my_static,5,$eof;// As of PHP 5.3.0
  21.  
  22. echo SonFoo::$my_static,6,$eof;
  23.  
  24. $sonfoo = new SonFoo();
  25. echo $sonfoo->sonStatic(),7,$eof;
  26.  
  27. }
  28. }

调用

  1. UseFoo::index();

结果:

foo1
foo2

( ! ) Strict standards: Accessing static property Libs\Foo::$my_static as non static in ......Libs\UseFoo.php on line 15
Call Stack
# Time Memory Function Location
1 0.0008 244064 {main}( ) ..\index.php:0
2 0.0026 262624 Libs\UseFoo::index( ) ..\index.php:51
( ! ) Notice: Undefined property: Libs\Foo::$my_static in ......Libs\UseFoo.php on line15
Call Stack
# Time Memory Function Location
1 0.0008 244064 {main}( ) ..\index.php:0
2 0.0026 262624 Libs\UseFoo::index( ) ..\index.php:51

3
foo4
foo5
foo6
foo7

  • 使用静态变量:

  变量范围的另一个重要特性是静态变量(static variable)。静态变量仅在局部函数域中存在,但当程序执行离开此作用域时,其值并不丢失。看看下面的例子:

  1. <?php
  2. function test($id)
  3. {
  4. static $count=0;
  5. echo "{$id}=>count is :".$count."<br/>";
  6. $count++;
  7. }
  8.  
  9. test(1);
  10. test(2);
  11. test(3);

结果:

  1. 1=>count is 0
  2. 2=>count is 1
  3. 3=>count is 2

  静态变量也提供了处理递归函数(递归函数是一种调用自己的函数)的方法。下面列出一个使用static变量完成递归的函数。

举个小例子

  1. function test($end)
  2. {
  3. static $sum=0;
  4. if($end>0){
  5. $sum +=$end;
  6. $end--;
  7. test($end);
  8. }
  9. return $sum;
  10. }
  11. echo test(10);//输出55

注意注意注意!上面代码只是一个小例子,实际生产中千万不要这么来求和!!!

  • 后期静态绑定:

  自 PHP 5.3.0 起,PHP 增加了一个叫做后期静态绑定的功能,用于在继承范围内引用静态调用的类。

  准确说,后期静态绑定工作原理是存储了在上一个“非转发调用”(non-forwarding call)的类名。当进行静态方法调用时,该类名即为明确指定的那个(通常在 :: 运算符左侧部分);当进行非静态方法调用时,即为该对象所属的类。所谓的“转发调用”(forwarding call)指的是通过以下几种方式进行的静态调用:self::,parent::,static:: 以及 forward_static_call()。可用 get_called_class() 函数来得到被调用的方法所在的类名,static:: 则指出了其范围。

  该功能从语言内部角度考虑被命名为“后期静态绑定”。“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为“静态绑定”,因为它可以用于(但不限于)静态方法的调用。

  1).self:: 使用self::或者__CLASS__对之前类的静态引用,取决于定义当前方法所在的类

  1. class A {
  2. public static function who() {
  3. echo __CLASS__;
  4. }
  5. public static function test() {
  6. self::who();
  7. }
  8. }
  9.  
  10. class B extends A {
  11. public static function who() {
  12. echo __CLASS__;
  13. }
  14. }
  15.  
  16. B::test();//输出A

  2).static:: 后期静态绑定本想通过引入一个新的关键字表示运行时最初调用的类来绕过限制。简单地说,这个关键字能够让你在上述例子中调用 test() 时引用的类是 B 而不是 A。最终决定不引入新的关键字,而是使用已经预留的 static 关键字。

  1. //简单用法
    class A {
  2. public static function who() {
  3. echo __CLASS__;
  4. }
  5. public static function test() {
  6. static::who(); // 后期静态绑定从这里开始
  7. }
  8. }
  9.  
  10. class B extends A {
  11. public static function who() {
  12. echo __CLASS__;
  13. }
  14. }
  15.  
  16. B::test();//输出B

  在非静态环境下,所调用的类即为该对象实例所属的类,由于$this-> 会在同一作用范围内尝试调用私有方法,而static::则可能给出不同结果。static::只能用于静态属性

  1. <?php
  2. class A {
  3. private function foo() {
  4. echo "success!\n";
  5. }
  6. public function test() {
  7. $this->foo();
  8. static::foo();
  9. }
  10. }
  11.  
  12. class B extends A {
  13. /* foo() will be copied to B, hence its scope will still be A and
  14. * the call be successful */
  15. }
  16.  
  17. class C extends A {
  18. private function foo() {
  19. /* original method is replaced; the scope of the new one is C */
  20. }
  21. }
  22.  
  23. $b = new B();
  24. $b->test();
  25. $c = new C();
  26. $c->test(); //fails
  27. ?>
  1. success!
  2. success!
  3. success!
  4.  
  5. Fatal error: Call to private method C::foo() from context 'A' in /tmp/test.php on line 9

后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止。另一方面,如果静态调用使用 parent:: 或者 self:: 将转发调用信息。

  1. class A {
  2. public static function foo() {
  3. static::who();
  4. }
  5.  
  6. public static function who() {
  7. echo __CLASS__."\n";
  8. }
  9. }
  10.  
  11. class B extends A {
  12. public static function test() {
  13. A::foo();
  14. parent::foo();
  15. self::foo();
  16. }
  17.  
  18. public static function who() {
  19. echo __CLASS__."\n";
  20. }
  21. }
  22. class C extends B {
  23. public static function who() {
  24. echo __CLASS__."\n";
  25. }
  26. }
  27.  
  28. C::test();//输出结果为ACC

推荐阅读>>

abstract class和interface的区别

abstract class和interface有什么区别?

深入分析PHP final关键字使用技巧

PHP抽象类

PHP 抽象方法与抽象类abstract关键字介绍及应用

PHP abstract final static 关键字

PHP 抽象类,接口,抽象方法,静态方法的更多相关文章

  1. PHP中的抽象类与抽象方法/静态属性和静态方法/PHP中的单利模式(单态模式)/串行化与反串行化(序列化与反序列化)/约束类型/魔术方法小结

      前  言  OOP  学习了好久的PHP,今天来总结一下PHP中的抽象类与抽象方法/静态属性和静态方法/PHP中的单利模式(单态模式)/串行化与反串行化(序列化与反序列化). 1  PHP中的抽象 ...

  2. java抽象类、抽象方法、接口、实现接口详解

    对于java中的抽象类,抽象方法,接口,实现接口等具体的概念就不在这里详细的说明了,网上书本都有很多解释,主要是我懒,下面通过一个例子来说明其中的精髓要点,能不能练成绝世武功,踏上封王之路,就看自己的 ...

  3. PHP面向对象(抽象类与抽象方法、接口的实现)

    一.抽象类与抽象方法 1,任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的. 2,定义为抽象的类不能被实例化. 3, 被定义为抽象的方法只是声明了其调用方式(参数) ...

  4. PHP的抽象类和抽象方法以及接口总结

    PHP中的抽象类和抽象方法自己用的不多,但是经常会在项目中看到别人使用,同样,今天在看别人的代码的时候,发现使用了抽象类,就总结下: 抽象类:1.如果一个类中有一个方法是抽象方法,则这个类就是抽象类: ...

  5. 速战速决 (4) - PHP: 类基础, 抽象类, 接口, trait

    [源码下载] 速战速决 (4) - PHP: 类基础, 抽象类, 接口, trait 作者:webabcd 介绍速战速决 之 PHP 类基础 抽象类 接口 trait 示例1.类的相关知识点 1(基础 ...

  6. 抽象类 接口 虚函数(C++模拟,个人见解)

    1.抽象类里面可以有非抽象方法但接口里只能有抽象方法声明方法的存在而不去实现它的类被叫做抽像类(abstract class),它用于要创建一个体现某些基本行为的类,并为该类声明方法,但不能在该类中实 ...

  7. 【Java基础】【09面向对象_多态&抽象类&接口】

    09.01_面向对象(多态的概述及其代码体现) A:多态(polymorphic)概述 事物存在的多种形态 B:多态前提 a:要有继承关系. b:要有方法重写. c:要有父类引用指向子类对象. C:案 ...

  8. -1-2 java 面向对象基本概念 封装继承多态 变量 this super static 静态变量 匿名对象 值传递 初始化过程 代码块 final关键字 抽象类 接口 区别 多态 包 访问权限 内部类 匿名内部类 == 与 equal

    java是纯粹的面向对象的语言 也就是万事万物皆是对象 程序是对象的集合,他们通过发送消息来相互通信 每个对象都有自己的由其他的对象所构建的存储,也就是对象可以包含对象 每个对象都有它的类型  也就是 ...

  9. python 全栈开发,Day21(抽象类,接口类,多态,鸭子类型)

    一.昨日复习 派生方法和派生属性 super 只有在子父类拥有同名方法的时候, 想使用子类的对象调用父类的方法时,才使用super super在类内 : super().方法名(arg1,..) 指名 ...

  10. 06 面向对象:多态&抽象类&接口&权限修饰符&内部类

    多态: /* 多态(polymorphic)概述 * 事物存在的多种形态 多态前提 * a:要有继承关系. * b:要有方法重写. * c:要有父类引用指向子类对象. * 成员变量 * 编译看左边(父 ...

随机推荐

  1. B -- POJ 1208 The Blocks Problem

    参考:https://blog.csdn.net/yxz8102/article/details/53098575 https://www.cnblogs.com/tanjuntao/p/867892 ...

  2. java web相对路径和绝对路径总结

    java web 开发过程中很多地方涉及url路径的问题,比如jsp页面.servlet之间的跳转.其实,可以将url中的/xxx看成一级目录,然后像看待目录层级之间的关系那样去看待url路径.接下来 ...

  3. 笔记-twisted-adbapi-scrapy

    笔记-twisted-adbapi-scrapy-mysql 1.      异步插入mysql 在爬虫中需要insert到mysql,但有一个问题是在爬虫环境中commit的及时性与性能冲突. 一般 ...

  4. js 实现字符串转日期进行比较大小

    代码如下 var a = '2016-01-01 12:12:12'; var b = '2016-01-01 12:12:13'; var al = new Date(a).getTime(); v ...

  5. Ubuntu 首次给root用户设置密码

    用过ubuntu的人都知道,刚安装好root用户是没有密码的,没有密码我们就没法用root用户登录.给root用户设置密码输入命令sudo passwd root,然后系统会让你输入密码,这时输入的密 ...

  6. url解读

    我刚刚学习的时候,我抓到包不知道哪个是协议.哪个是是服务器地址.哪个是端口号...不知道有没有老铁遇到跟我一样的. 接口:http://172.168.12.0:8888/old/login.do 解 ...

  7. eclipse返回快捷键

    1.图上第一个箭头(Ctrl + Q) 返回上一个编辑点(编辑,修改代码) 2.图上第二个箭头(Alt + Left) 返回上一个操作点(点击进入方法等操作) 3.图上第三个箭头(Alt + Righ ...

  8. Xshell6连接虚拟机(一)

    普通用户转换成管理员: 一.首先进入终端: 1.输入:    su   然后回车 2.若输入密码 则显示认证失败,说明还是普通管理员身份. (1)设置新密码:        sudo passwd r ...

  9. DFS(8)——poj2034Anti-prime Sequences

    一.题目回顾 题目链接:Anti-prime Sequences Sample Input 1 10 2 1 10 3 1 10 5 40 60 7 0 0 0   Sample Output 1,3 ...

  10. 微信小程序-腾讯地图显示偏差问题

    原文地址: http://fanjiajia.cn/2018/08/30/%E5%BE%AE%E4%BF%A1%E5%B0%8F%E7%A8%8B%E5%BA%8F-%E8%85%BE%E8%AE%A ...