什么是PHP中的抽象类?
什么是PHP中的抽象类?
如何使用?
抽象类是包含至少一个抽象方法的类,该方法中没有任何实际代码,只有名称和参数,并且已标记为“抽象”。
这样做的目的是提供一种要从中继承的模板,并强制继承类实现抽象方法。
因此,抽象类介于常规类和纯接口之间。接口也是抽象类的一种特殊情况,其中所有方法都是抽象的。
有关进一步的参考,请参阅 PHP 手册的这一部分。
抽象类是包含一个或多个抽象方法的类。抽象方法是已声明但不包含任何实现的方法。抽象类可能未实例化,并且需要子类来为抽象方法提供实现。
1. 无法实例化抽象类:定义为抽象的类不得实例化,任何包含至少一个抽象方法的类也必须是抽象的。
示例如下:
abstract class AbstractClass
{
abstract protected function getValue();
abstract protected function prefixValue($prefix);
public function printOut() {
echo "Hello how are you?";
}
}
$obj=new AbstractClass();
$obj->printOut();
//Fatal error: Cannot instantiate abstract class AbstractClass
2. 任何包含至少一个抽象方法的类也必须是抽象的:抽象类可以有抽象方法和非抽象方法,但它必须至少包含一个抽象方法。如果一个类至少有一个抽象方法,则必须将该类声明为 abstract。
注意:特征支持使用抽象方法,以便对参展类施加要求。
示例如下:
class Non_Abstract_Class
{
abstract protected function getValue();
public function printOut() {
echo "Hello how are you?";
}
}
$obj=new Non_Abstract_Class();
$obj->printOut();
//Fatal error: Class Non_Abstract_Class contains 1 abstract method and must therefore be declared abstract or implement the remaining methods (Non_Abstract_Class::getValue)
3. 抽象方法不能包含 body:定义为抽象的方法只是声明方法的签名 - 它们不能定义实现。但是非抽象方法可以定义实现。
abstract class AbstractClass
{
abstract protected function getValue(){
return "Hello how are you?";
}
public function printOut() {
echo $this->getValue() . "\n";
}
}
class ConcreteClass1 extends AbstractClass
{
protected function getValue() {
return "ConcreteClass1";
}
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass1";
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue('FOO_') ."\n";
//Fatal error: Abstract function AbstractClass::getValue() cannot contain body
4. 从抽象类继承时,父类声明中标记为抽象的所有方法都必须由子类定义:如果继承抽象类,则必须为其中的所有抽象方法提供实现。
abstract class AbstractClass
{
// Force Extending class to define this method
abstract protected function getValue();
// Common method
public function printOut() {
print $this->getValue() . "<br/>";
}
}
class ConcreteClass1 extends AbstractClass
{
public function printOut() {
echo "dhairya";
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
//Fatal error: Class ConcreteClass1 contains 1 abstract method and must therefore be declared abstract or implement the remaining methods (AbstractClass::getValue)
5. 相同(或限制较少)的可见性:从抽象类继承时,父类声明中标记为抽象的所有方法必须由子类定义;此外,这些方法必须以相同(或限制较少)的可见性进行定义。例如,如果抽象方法被定义为受保护,则必须将函数实现定义为受保护或公共,但不是私有。
请注意,抽象方法不应是私有的。
abstract class AbstractClass
{
abstract public function getValue();
abstract protected function prefixValue($prefix);
public function printOut() {
print $this->getValue();
}
}
class ConcreteClass1 extends AbstractClass
{
protected function getValue() {
return "ConcreteClass1";
}
public function prefixValue($prefix) {
return "{$prefix}ConcreteClass1";
}
}
$class1 = new ConcreteClass1;
$class1->printOut();
echo $class1->prefixValue('FOO_') ."<br/>";
//Fatal error: Access level to ConcreteClass1::getValue() must be public (as in class AbstractClass)
6. 抽象方法的签名必须匹配:从抽象类继承时,父类声明中所有标记为抽象的方法必须由子级定义;方法的签名必须匹配,即类型提示和所需参数的数量必须相同。例如,如果子类定义了一个可选参数,而抽象方法的签名没有,则签名中没有冲突。
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"), "<br/>";
echo $class->prefixName("Pacwoman"), "<br/>";
//output: Mr. Pacman
// Mrs. Pacwoman
7. 抽象类不支持多重继承:抽象类可以扩展另一个抽象类,抽象类可以提供接口的实现。但它不支持多重继承。
interface MyInterface{
public function foo();
public function bar();
}
abstract class MyAbstract1{
abstract public function baz();
}
abstract class MyAbstract2 extends MyAbstract1 implements MyInterface{
public function foo(){ echo "foo"; }
public function bar(){ echo "bar"; }
public function baz(){ echo "baz"; }
}
class MyClass extends MyAbstract2{
}
$obj=new MyClass;
$obj->foo();
$obj->bar();
$obj->baz();
//output: foobarbaz
注意:请注意,代码中类的顺序或位置可能会影响解释器,并可能导致致命错误。因此,当使用多个抽象级别时,请注意类在源代码中的位置。
下面的示例将导致致命错误:找不到类“马”
class cart extends horse {
public function get_breed() { return "Wood"; }
}
abstract class horse extends animal {
public function get_breed() { return "Jersey"; }
}
abstract class animal {
public abstract function get_breed();
}
$cart = new cart();
print($cart->get_breed());