首先呢,小编在这里先给大家解释一下啥叫抽象类吧。
学过C++语言的宝宝们应该对抽象类都很熟悉吧,其实呢,抽象类这个概念是不会随着语言种类的变化而变化的。任何一个类,如果它里面至少有一个方法是被声明为抽象的,那么这个类就必须被声明为抽象的,这就是抽象类。
下面是一些抽象类的使用规则
- 定义为抽象的类不能被实例化。
- 被定义为抽象的方法只是声明了其调用方式(参数),不能定义其具体的功能实现。
- 继承一个抽象类的时候,子类必须定义父类中的所有抽象方法,这些方法的访问控制必须和父类中一样(或者更为宽松)。
- 访问控制由紧到松的顺序依次为:private、protected、public。
好啦,就让我们通过实例来进一步的了解抽象类吧
首先请大家分析一下下面的代码
<?php
abstract class AbstractClass
{
// 强制要求子类定义这些方法
abstract protected function getValue();
abstract protected function prefixValue($prefix);
// 普通方法(非抽象方法),子类可选择性继承
public function printOut() {
print $this->getValue() . PHP_EOL;
}
}
class ConcreteClass1 extends AbstractClass
{
protected function getValue() {
return "ConcreteClass1";
}
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass1"; //注意一下这里的输出方式哦,和下面进行比较可能会有收获
}
}
class ConcreteClass2 extends AbstractClass
{
public function getValue() {
return "ConcreteClass2";
}
public function prefixValue($prefix) {
return "$prefix"." ConcreteClass2"; // 就是这里哦,不一样的输出方式,会产生不一样的效果,当然也可以产生相同的效果,自己试试吧
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue('FOO_LALALA') . PHP_EOL;
$class2 = new ConcreteClass2;
$class2->printOut();
echo $class2->prefixValue('FOO_WAWAWA') . PHP_EOL;
?>
输出结果如下:
ConcreteClass1
FOO_LALALAConcreteClass1
ConcreteClass2
FOO_WAWAWA ConcreteClass2
小编提示一下大家,注意abstract关键字在类和函数(方法)使用时的位置,特别注意一下子类在继承父类的抽象函数时,函数的访问控制规则。还有啊,子类中的函数时不带有abstract关键字的哦~
下面是抽象函数的一种更为灵活的一种用法
<?php
abstract class AbstractClass
{
// 我们的抽象方法仅需要定义需要的参数
abstract protected function prefixName($name);
}
class ConcreteClass extends AbstractClass
{
// 我们的子类可以定义父类签名中不存在的可选参数
public function prefixName($name, $separator = ".") { //在这里多加了一个参数
if ($name == "Pacman") {
$prefix = "Mr";
} elseif ($name == "Pacwoman") {
$prefix = "Mrs";
} else {
$prefix = "";
}
return "{$prefix}{$separator} {$name}";
}
}
$class = new ConcreteClass;
echo $class->prefixName("Pacman"), "\n";
echo $class->prefixName("Pacwoman"), "\n";
?>
运行结果如下:
Mr. Pacman
Mrs. Pacwoman
大家可以看到,在这个例子中,子类在继承父类中的抽象函数时多加了一个参数,注意哈,在这里是允许的。