测试环境
PHP version 5.3.29/5.6.29/7.1.0
Apache 2.4.25
PHP 对待对象的方式与引用和句柄相同,即每个变量都持有对象的引用,而不是整个对象的拷贝。
自 PHP 5.5 起,关键词
class
也可用于类名的解析。使用 ClassName::class 可以获取一个字符串,包含了类 ClassName 的完全限定名称,这对使用了 命名空间 的类尤其有用。
namespace NS {
class ClassName {
}
echo ClassName::class; //outputs NS\ClassName
}
属性和方法位于不同的“存储空间”,这就意味着不能直接调用一个赋值给属性的匿名方法,需要先把属性赋值给一个变量。
PHP 7.0.0 之后则可以将该属性用
()
括起来然后直接调用该匿名方法。
class Foo{
public $bar;
public function __construct() {
$this->bar = function() {
return 2;
};
}
}
$obj = new Foo();
// as of PHP 5.3.0:
$func = $obj->bar;
echo $func(), PHP_EOL;
// alternatively, as of PHP 7.0.0:
echo ($obj->bar)(), PHP_EOL;
PHP 5.4 引进了一种新的创建对象后调用成员方法的表达式。
echo (new DateTime())->format('Y');
属性常量的初始化值必须是常量,PHP 5.6 之后也可以是常量表达式。
class SimpleClass{
//PHP 5.6.0+有效:
public $var1 = 'hello ' . 'world';
//PHP 5.3.0+有效:
public $var2 = <<
//PHP 5.6.0+有效:
public $var3 = 1+2;
// 无效的属性声明:
// public $var4 = self::myStaticMethod();
// public $var5 = $myVar;
// 有效的属性声明:
public $var6 = myConstant;
public $var7 = array(true, false);
//PHP 5.3.0+有效:
public $var8 = <<<'EOD'
hello world
EOD;
}
$this
可以强转成数组。如果一个 object 类型转换为 array,则结果为一个数组,其单元为该对象的属性。键名将为成员变量名,不过有几点例外:静态属性不可访问;私有变量前会加上类名作前缀;受保护变量前会加上一个 ‘*’ 做前缀。这些前缀的前后都各有一个 NULL 字符(\0)。
class TestA {
public $var1 = 1;
protected $var2 = 2;
private $var3 = 3;
static $var4 = 4;
public function toArray() {
return (array) $this;
}
}
class TestB extends TestA{
public $var5 = 5;
protected $var6 = 6;
private $var7 = 7;
static $var8 = 8;
}
$t = new TestB;
print_r($t->toArray());
与其它方法不同,当__construct()
被与父类__construct()
具有不同参数的方法覆盖时,PHP 不会产生一个 E_STRICT 错误信息。即构造方法可以重载。
尝试从析构方法中抛出异常会导致fatal error,然而以下代码在PHP 7.1.0上会导致出错,在PHP 5.3.29 和PHP 5.6.29都能正常运行。
class Something {
public function __construct() {
echo __METHOD__ . '
';
}
public function __destruct(){
echo __METHOD__ .'
';
throw new Exception("Exception thrown out of ::__destruct()", 1);
}
}
try{
$Something = new Something();
$Something = null; // will cause to call the objects ::__destruct()
// also possible: unset($Something);
}catch(Exception $e){
echo 'Exception: ' . $e->getMessage() . PHP_EOL.'
';
}
echo 'End of script -- no Fatal Error.';
对属性或方法的访问控制,是通过在前面添加关键字 public(公有),protected(受保护)或 private(私有)来实现的。
被定义为公有的类成员可以在任何地方被访问。
被定义为受保护的类成员则可以被其自身以及其子类和父类访问。
被定义为私有的类成员则只能被其定义所在的类访问。
上述说法并不严谨,通过
call_user_func()
和Closure::bind()
复制一个指定的绑定对象和类作用域的闭包,可以在类外访问类成员内部的私有成员和受保护成员。
class Sealed {
private $value = 'foo';
}
$sealed = new Sealed;
var_dump($sealed); // private $value => string(3) "foo"
// call_user_func — 把第一个参数作为回调函数调用。
// PHP5.4 Closure::bind — 复制一个闭包,绑定指定的$this对象和类作用域。
call_user_func(\Closure::bind(
function () use ($sealed) { $sealed->value = 'BAZ'; },
null,
$sealed
));
var_dump($sealed); // private $value => string(3) "BAZ"
PHP 5.3 之后可以使用简单的变量动态调用类的静态方法或常量,但是使用对象属性调用类的静态方法或类常量,在PHP 7.0之前的版本会抛出语法错误。
class Foo{
const BAR = 'const';
public static function bar(){
return 'static';
}
}
echo Foo::BAR ; // outputs "const"
echo Foo::bar() ; // outputs "static"
$var = 'Foo';
echo $var::BAR ; // outputs "const" in PHP 5.3.0
echo $var::bar(); // outputs "static" in PHP 5.3.0
//PHP 7.0 之后有效
$object = new stdClass;
$object->class = 'Foo';
echo $object->class::BAR . PHP_EOL; // outputs "const" in PHP 7
echo $object->class::bar() . PHP_EOL; // outputs "static" in PHP 7
不能在抽象类的内部通过self实现抽象类内部的抽象方法。
abstract class Basic {
public static function doWork() {
return (new self())->work();
}
abstract protected function work();
}
class Keeks extends Basic {
protected function work() {
return 'Keeks';
}
}
echo Keeks::doWork();
继承一个抽象类的时候,子类必须定义父类中的所有抽象方法;另外,这些方法的访问控制必须和父类中一样(或者更为宽松)。
此外抽象方法的调用方式必须匹配,即类型和所需参数数量必须一致。(如果子类参数列表中有默认参数则例外)
// fatal error,子类方法的调用方法必须和抽象父类的类型和数量一致
abstract class my_class {
abstract public function my_function($number);
}
class subclass extends my_class {
//now $string isn't initialized
public function my_function($new_number, $string ){
echo $new_number . $string;
}
}
$var = new subclass();
$var->my_function(1024, ' is an integer!!!');
子类可以定义父类抽象方法中不存在的可选参数。
// 子类的方法调用必须和抽象类的类型和参数一致
// 但是子类定义的方法参数和数量可以和父类的不一致(如子类设置了默认参数)
abstract class my_class {
abstract public function my_function($number);
}
class subclass extends my_class {
public function my_function($new_number, $string = ' is an integer!!!'){
echo $new_number . $string;
}
}
$var = new subclass();
$var->my_function(1024);
子类的方法调用可以和抽象父类的类型和参数不一致,如重写了子类的默认参数。
// 子类的方法调用可以和抽象父类的类型和参数不一致
// 子类定义的方法参数和数量可以和父类的不一致(如重写了子类的默认参数)
abstract class my_class {
abstract public function my_function($number);
}
class subclass extends my_class {
public function my_function($new_number, $string = ' is a float!!!'){
echo $new_number . $string;
}
}
$var = new subclass();
//added ' is an integer'
$var->my_function(1024, ' is an integer!!!');
抽象类的抽象方法可以是静态的,但最好不要这么做。虽然在PHP 7.1上无错误,但是在PHP5.6.29上提示Strict standards:静态方法不能用在抽象类里。这是因为静态方法只能由该类调用,但是抽象类意味着一定要有其他的类去实现它的方法。
虽然抽象类不能被实例化,但是抽象类可以调用类内的静态非抽象方法。
abstract class FOO{
abstract static function dump();
static function bar(){
echo "test\n";
}
}
class BAR extends FOO{
static function dump(){var_dump("BAR");}
}
BAR::dump(); // string(3) "BAR"
FOO::bar(); // test
抽象类不一定是基类,也可以继承自非抽象类。
class Foo {
public function sneeze() { echo 'achoooo','
'; }
}
abstract class Bar extends Foo {
public abstract function hiccup();
}
class Baz extends Bar {
public function hiccup() { echo 'hiccup!','
'; }
}
$baz = new Baz();
$baz->sneeze(); // achoooo
$baz->hiccup(); //hiccup!
implements
操作符。类中必须实现接口中定义的所有方法,否则会报一个致命错误。extends
操作符。实现接口的方法时,如果方法的参数中设置了默认值,则可以和接口中定义的方法不一致,且不会报错。
interface Foo {
public function foo($foo);
}
class Bar implements Foo {
public function foo($foo, $bar = null) {
return $foo . $bar;
}
}
$bar = new Bar();
echo $bar->foo('Study', 'Good');
接口中的常量不能被子类或子接口重写,但是可以使用一个(抽象)类实现该接口,然后另一个类继承实现该接口的类,就可以重写接口常量。
interface inter1 {
const interface1 = "I am from interface 1";
}
abstract class AbsClass implements inter1 {
}
class Test extends AbsClass {
const interface1 = "I am from test class";
public function display() {
echo inter1::interface1;
echo "
";
echo Test::interface1;
}
}
$Obj = new Test();
$Obj->display();
可以在类声明之前实例化该类的对象,但是并不能在实现接口之前实例化实现接口的类。
$bar = new FooA(); // 有效
class FooA{}
// $bar = new FooB(); // 无效 - 抛出致命错误,找不到类FooB
interface foo2{}
class FooB implements foo2{}
$bar = new FooB();
自 PHP 5.4.0 起,PHP 实现了一种代码复用的方法,称为 trait。Trait 为了减少单继承语言的限制,使开发人员能够自由地在不同层次结构内独立的类中复用 method。详见手册
从基类继承的成员会被 trait 插入的成员所覆盖。优先顺序是来自当前类的成员覆盖了 trait 的方法,而 trait 则覆盖了被继承的方法。
trait Hello {
public function sayHello() {
echo 'Hello ';
}
}
trait World {
public function sayWorld() {
echo 'World';
}
}
class MyHelloWorld {
use Hello, World; // 通过逗号分隔,在 use 声明列出多个 trait
public function sayExclamationMark() {
echo '!';
}
}
$o = new MyHelloWorld();
$o->sayHello();
$o->sayWorld();
$o->sayExclamationMark();
如果 trait 定义了一个属性,那类将不能定义同样名称的属性,否则会产生一个错误。如果该属性在类中的定义与在 trait 中的定义兼容(同样的可见性和初始值)则错误的级别是 E_STRICT,否则是一个致命错误。
trait PropertiesTrait {
public $same = true;
public $different = false;
}
class PropertiesExample {
use PropertiesTrait;
public $same = true; // Strict Standards
public $different = true; // 致命错误
}
PHP 7 开始支持匿名类。可以传递参数到匿名类的构造器,也可以继承(extends)其他类、实现接口(implement s),以及像其他普通的类一样使用 trait。
class SomeClass {}
interface SomeInterface {}
trait SomeTrait {}
var_dump(new class(10) extends SomeClass implements SomeInterface {
private $num;
public function __construct($num){
$this->num = $num;
}
use SomeTrait;
});
匿名类由引擎分配名字,如下所示:
echo get_class(new class {}); // class@anonymous G:\www\php\\test.php00480021
PHP所提供的”重载”(overloading)是指动态地”创建”类属性和方法。我们是通过魔术方法(magic methods)来实现的。
当调用当前环境下未定义或不可见的类属性或方法时,重载方法会被调用。
所有的重载方法都必须被声明为 public。
程序运行时已经存在类的实例化对象的属性,但是被unset()之后,也可以实现重载。
class Test {
public $property1;
public function __get($name) {
return "Get called for " . get_class($this) . "->\$$name \n";
}
}
$Test = new Test();
unset($Test->property1); // enable overloading
echo $Test->property1; // Get called for Test->$property1
可以通过魔术方法在类外获取私有属性。但是不应该在程序运行时动态改变属性值。
class order{
private $OrderID='';
private $OrderAmount=0.00;
public function __set($name='', $value=''){
if(property_exists($this, $name)){
$this->$name = $value;
}
}
public function __get($name=''){
$value = null;
if(property_exists($this, $name)){
$value = $this->$name;
}
return $value;
}
}
$order = new order();
$order->OrderID = '201305062202';
$order->OrderAmount = 23.45;
$order->InvalidMember = 'Missed Assignment';
echo ''
, print_r($order, true), '
';
调用对象不存在的方法时请区分静态方法和非静态方法,静态方法用
__callstatic()
调用,非静态方法用__call()
调用。而且非静态方法只能通过类的实例化对象调用,否则会报错。
class TestClass {
function __call($method, $args) {
echo "Non-Static Method {$method} with args: " . print_r($args, TRUE);
}
static function __callstatic($method, $args) {
echo "Static Method {$method} with args: " . print_r($args, TRUE);
}
}
$obj = new TestClass();
$obj->method_doesnt_exist('Non-static Method');
echo "
";
TestClass::method_doesnt_exist('Static Method');
如果父类中的方法被声明为 final,则子类无法覆盖该方法。如果一个类被声明为 final,则不能被继承。
属性不能被定义为 final,只有类和方法才能被定义为 final。
类的私有成员不能被继承,所以子类中可以定义父类的同名方法,但是如果父类定义了
private final
方法,子类中一定不要出现与同类同名的方法,否则会报错。
class A {
private final function method(){}
}
class B extends A {
private function method(){}
}
“后期绑定”的意思是说,static:: 不再被解析为定义当前方法所在的类,而是在实际运行时计算的。也可以称之为”静态绑定”,因为它可以用于(但不限于)静态方法的调用。
后期静态绑定的解析会一直到取得一个完全解析了的静态调用为止。另一方面,如果静态调用使用
parent::
或者self::
将转发调用信息。
class A {
public static function foo() {
static::who();
}
public static function who() {
echo __CLASS__."\n";
}
}
class B extends A {
public static function test() {
A::foo();
parent::foo();
self::foo();
}
public static function who() {
echo __CLASS__."\n";
}
}
class C extends B {
public static function who() {
echo __CLASS__."\n";
}
}
C::test(); // outputs A C C
B::test(); // outputs A B B
PHP 的引用是别名,就是两个不同的变量名字指向相同的内容。在php5,一个对象变量只是保存一个标识符来访问真正的对象内容。
当对象作为参数传递,作为结果返回,或者赋值给另外一个变量,另外一个变量跟原来的不是引用的关系,只是他们都保存着同一个标识符的拷贝,这个标识符指向同一个对象的真正内容。
请注意unset删除对象和给对象赋值为null的区别。
class A {
public $foo = 1;
}
$a = new A;
$b = $a;
$a->foo = 2;
$a = NULL;
echo $b->foo."
\n"; // 2
$c = new A;
$d = &$c;
$c->foo = 2;
$c = NULL;
echo $d->foo."
\n"; // Notice: Trying to get property of non-object...
$e = new A;
$f = &$e;
$e ->foo = 2;
unset($e);
echo $f->foo."
\n"; // 2
所有php里面的值都可以使用函数serialize()来返回一个包含字节流的字符串来表示。unserialize()函数能够重新把字符串变回php原来的值。 序列化一个对象将会保存对象的所有变量,但是不会保存对象的方法,只会保存类的名字。
对象的静态成员不能被序列化,
protected
类型的受保护属性前加*
做前缀,private
类型的私有属性前加类名做前缀,反序列化后只保存对象的变量,不保留方法。
// file "classa.php"
class A{
public $one = 1;
static $two = 2; //静态成员并没有存储在序列化文件中
protected $three =3;
private $four = 4;
public function show_one(){
echo $this ->one;
}
public function show_two(){
echo self::$two;
}
}
// file "page1.php"
include "classa.php";
$a = new A;
$s = serialize($a);
file_put_contents('store.txt', $s);
echo $s;
//O:1:"A":3:{s:3:"one";i:1;s:8:"*three";i:3;s:7:"Afour";i:4;}
// file "page2.php"
include 'classa.php';
$s = file_get_contents('store.txt');
$a = unserialize($s);
$a -> show_one();
$a -> show_two();
var_dump($a) ;
/* object(A)#1 (3) {
["one"]=> int(1)
["three":protected]=> int(3)
["four":"A":private]=> int(4)
}
*/