抽象类
定义为抽象的类不能被实例化。
任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的。
被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。
继承一个抽象类的时候,子类必须定义父类中的所有抽象方法,这些方法的访问控制必须和父类中一样(或者更为宽松)。
语法
<?php
abstract class AbstractClass
{
//抽象方法 -- 强制要求子类定义这些方法
// 我们的抽象方法仅需要定义需要的参数
abstract protected function getValue();
abstract protected function prefixValue($prefix);
abstract protected function prefixName($name);
//普通方法(非抽象)
public function printOut()
{
print $this->getValue() . "\n";
}
}
//子类继承抽象类
class ConcreteClass1 extends AbstractClass
{
protected function getValue() {
return "ConcreteClass1";
}
//访问权限可以比父类更宽松或同权限
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass1";
}
// 我们的子类可以定义父类签名中不存在的可选参数
public function prefixName($name, $separator = ".") {
if ($name == "Pacman") {
$prefix = "Mr";
} elseif ($name == "Pacwoman") {
$prefix = "Mrs";
} else {
$prefix = "";
}
return "{$prefix}{$separator} {$name}";
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue('FOO_') ."\n";
echo $class1->prefixName("Pacman"), "\n";
//以上例程会输出:
//ConcreteClass1
//FOO_ConcreteClass1
//Mr. Pacman
接口(对象接口)
使用接口 (interface) 关键字,可以指定某个类必须实现哪些方法,但不需要定义这些方法的具体内容,就像定义一个标准的类一样,但其中定义所有的方法都是空的。
接口中定义的所有方法都必须是公有,这是接口的特性。
在接口中定义一个构造方法是被允许的。在有些场景下这可能会很有用,例如用于工厂模式时。
接口中也可以定义常量。接口常量和类常量的使用完全相同,但是不能被子类或子接口所覆盖。
要实现一个接口,使用 implements
操作符。类中必须实现接口中定义的所有方法,否则会报一个致命错误。
类可以实现多个接口,用逗号来分隔多个接口的名称。(注意:在 PHP 5.3.9 之前,实现多个接口时,接口中的方法不能有重名,因为这可能会有歧义。在最近的 PHP 版本中,只要这些重名的方法签名相同,这种行为就是允许的。)
接口也可以继承,通过使用 extends 操作符。
类要实现接口,必须使用和接口中所定义的方法完全一致的方式。否则会导致致命错误。
语法
#实现接口
<?php
// 声明一个'iTemplate'接口
interface iTemplate
{
public function setVariable($name, $var);
public function getHtml($template);
}
// 实现接口
// 下面的写法是正确的
class Template implements iTemplate
{
private $vars = array();
public function setVariable($name, $var)
{
$this->vars[$name] = $var;
}
public function getHtml($template)
{
foreach($this->vars as $name => $value) {
$template = str_replace('{' . $name . '}', $value, $template);
}
return $template;
}
}
// 下面的写法是错误的,会报错,因为没有实现 getHtml():
// Fatal error: Class BadTemplate contains 1 abstract methods
// and must therefore be declared abstract (iTemplate::getHtml)
class BadTemplate implements iTemplate
{
private $vars = array();
public function setVariable($name, $var)
{
$this->vars[$name] = $var;
}
}
#可拓充的接口
<?php
interface a
{
public function foo();
}
interface b extends a
{
public function baz(Baz $baz);
}
// 正确写法
class c implements b
{
public function foo()
{
}
public function baz(Baz $baz)
{
}
}
// 错误写法会导致一个致命错误
class d implements b
{
public function foo()
{
}
public function baz(Foo $foo)
{
}
}
#继承多个接口
<?php
interface a
{
public function foo();
}
interface b
{
public function bar();
}
interface c extends a, b
{
public function baz();
}
class d implements c
{
public function foo()
{
}
public function bar()
{
}
public function baz()
{
}
}
#使用接口常量
<?php
interface a
{
const b = 'Interface constant';
}
// 输出接口常量
echo a::b;
// 错误写法,因为常量不能被覆盖。接口常量的概念和类常量是一样的。
class b implements a
{
const b = 'Class constant';
}
问答分析
- 抽象类和接口分别能直接实例化吗?
- 都不能直接实例化
- 抽象类的方法可以存在方法体吗?
- 抽象类中,定义为抽象方法的不可以存在方法体,抽象类中普通方法可以存在方法体。
- 抽象方法和普通方法的区别是什么?
- 抽象方法只是声明了其调用方式(参数),不能定义其具体的功能实现,普通方法可以定义具体的实现。
- 抽象方法可以是静态方法吗?
- 可以(PHP5+版本会出现严格模式提示,PHP7+无错,均可以运行)
- 抽象类可以存在成员属性吗?所有等级都可以吗?
- 可以存在成员属性,所有等级都可以,但是子类能直接调用的只有公开属性
- 接口的方法可以存在方法体吗?
- 接口方法不可以有方法体
- 接口可以存在成员属性吗?所有等级都可以吗?
- 不可以存在,接口不包括成员属性
- 普通类可以同时继承多个抽象类吗?如果可以如何实现?
- 抽象类不可以多继承 (多继承实现可以使用Trait)
- 普通类可以同时实现多个接口类吗?如果可以如何实现?
- 可以,类可以实现多个接口,用逗号来分隔多个接口的名称。
- 抽象类可以实现接口吗?如果可以,那抽象类需要实现接口定义的方法体吗?
- 抽象类可以实现接口,并且不需要实现接口的方法,但是普通类继承抽象类后需要全部实现
- 普通类可以既继承抽象类又实现接口吗?
- 可以