时间:2021-07-01 10:21:17 帮助过:29人阅读
class test { public function f1() { echo 'you call me!'; } public function f2() { f1(); }}$t = new test();$t->f2(); // 报错// public function f2() { // 正确// self::f1();// }// public function f2() { // 正确// static::f1();// }
一个值得注意的地方 __clone() cannot accept any arguments
class test { public function __construct() { echo 'ok'; } public function __destruct() { echo 'fail', '
'; }}$a = new test();$b = $a; // 指向同一个内存空间echo '
';// ok// -----// fail
class test { public function __construct() { echo 'ok'; } public function __destruct() { echo 'fail', '
'; }}$a = new test();$b = clone $a; // 指向不同的 内存空间echo '
';// ok// ---------------// fail// fail
class test { public function __construct() { echo 'ok', '
'; } public function __destruct() { echo 'fail', '
'; } public function __clone() { return self::__construct(); }}$a = new test();$b = clone $a; // 指向不同的 内存空间echo '
';// ok// ok// -----// fail// fail
class A { function foo() { if (isset($this)) { echo 'this is defined (', get_class($this), ')', '
'; } else { echo 'this is not defined', '
'; } }}class B { function bar() { A::foo(); }}$a = new A(); // this 变成 A$a->foo();A::foo();// this is defined (A)// this is not defined$b = new B(); // this 变成 B$b->bar();B::bar();// this is defined (B)// this is not defined
继承的时候权限只能越来越大, 不能变小
class A { protected $a = 'a'; protected function f() { echo $this->a, '
'; }}class B extends A{ private function f() { echo $this->a, '
'; }}$b = new B();$b->f(); // 报错
class A { protected $a = 'a'; public function f() { echo $this->a, '
'; }}class B extends A{ public function f() { echo $this->a, '
'; }}$b = new B(); // 可以继承 protected$b->f();
static 变量
所有实例化的实例化的类共享的变量, 其中一个改变了, 其他的也跟着改变
class stu { public static $fee; function f($fee) { self::$fee += $fee; }}
static 方法
不可以操作 $this 变量
, 使用 static 方法的情景, 一般的原则是某个方法中不包含 $this
变量
calss stu { static function func1() { static::func2(); } static function func2() { ... }}
调用: stu::func2();
, stu::func1();
使用 finnal 的场景:
final 不能修饰成员属性 (变量)
final 方法, 表示此方法不能被重写
class BaseClass { public function test() { echo "BaseClass::test() called\n"; } final public function moreTesting() { echo "BaseClass::moreTesting() called\n"; }}class ChildClass extends BaseClass { public function moreTesting() { echo "ChildClass::moreTesting() called\n"; }}// Results in Fatal error: Cannot override final method BaseClass::moreTesting()
final 类, 表示此类不能被继续被 extends
final class BaseClass { public function test() { echo "BaseClass::test() called\n"; } // 这里无论你是否将方法声明为 final,都没有关系 final public function moreTesting() { echo "BaseClass::moreTesting() called\n"; }}class ChildClass extends BaseClass {}// 产生 Fatal error: Class ChildClass may not inherit from final class (BaseClass)
class MyClass { const constant = 'constant value' ; function showConstant () { echo self :: constant . "\n" ; }}echo $class :: constant . "\n" ; // 自 PHP 5.3.0 起
延迟绑定的意思, self 的绑定是运行时候计算, 看下面
class Par { public static function whoami() { echo 'parent', '
'; } public static function say() { self::whoami(); } public static function say2() { static::whoami(); }}class Sun extends Par { public static function whoami() { echo 'sun', '
'; }}Sun::say(); // parent, 因为执行环境变成了 parentSun::say2(); // sun, 保持静态
class mysql { public static $ins; private function __construct() { /* do something */ } private function __clone () {} // 禁止继承的类修改自身, 也可以使用 final public static function getIns() { if (!(self::$ins instanceof mysql)) { self::$ins = new self(); } return self::$ins; }}$a = mysql::getIns();$b = mysql::getIns();var_dump ($a === $b); // true
class test { public $hello = 'normal hello'; public function __get($args) { if ($args == 'hello') { echo 'from get: hello'; } else if ($args = 'other') { echo 'from get: other'; } }}$t = new test();echo $t->hello, '
'; // normal hello, 如果存在不会调用 魔术方法echo $t->other; // 不能调用, 调用魔术方法
class test { public $hello = 'normal hello'; public function __set($k, $v) { echo 'you want to set me! ', $k, '
'; $this->$k = $v; }}$t = new test();$t->ttt = 222;echo $t->ttt; // 可以设置$t->hello = 'another hello';print_r($t);// you want to set me! ttt// 222// test Object// (// [hello] => another hello// [ttt] => 222// )
class test { public function __isset($v) { return false; ..... return true; }}
$t = new test();if (isset($t->value)) { echo 'isset';} else { echo 'noset';}
class test { public function func_empty() { echo 'nothing here'; } public function __call ($func, $args) { self::func_empty(); }}$t = new test();
class test { public function func_empty() { echo 'nothing here'; } public function __call ($func, $args) { print_r($args); self::func_empty(); }}$t = new test();$t->abc('abc', 'def');// Array ( [0] => abc [1] => def )
public static function __callStatic ( $name , $arguments ) { ... }
覆盖: override 指的是子类覆盖父类的方法, 在 PHP 中, 如果 子类和父类的参数不一样会有警告
重载: redeclare 指的是同一个类中包含两个相同名字的方法, PHP 不支持
实现重载的方法
func_num_args() 返回数据的参数的个数$arr = func_get_args() 让参数以数组的形式返回func_get_arg(0) 返回第一个参数func_get_arg(1) 返回第二个参数
public function f() { $arr = func_get_args(); foreach($arr as $v) { .... }}
__LINE____FILE____DIR____FUNCTION____CLASS____METHOD__
abstract
abstract class Car { abstract function func1();}class Fastcar extends Car { public function func1() { ... }}
interface
interface iUsb { public function f();}class phone implements iUsb { public function f() { .. }}
定义规范
当多个类, 之间平级的, 有相同的方法, 但是实现不同
类可以实现多个接口, 接口可以实现多个接口
class t1 implements i1, i2, i3 { }interface i1 extends i2, i2, i3 {}
class Monkey { public function climb() { ... }}interface iBirdable { public function fly();}interface iFishable { public function swim();}class NewMonkey extends Monkey implements iBirdable, iFishable { public function fly() { ... } public function swim() { ... }}
抽象类和接口的区别我不是 很熟悉
T_T, mark 个 todo 以后再来补充
如果文件 1 有 test
对象, 文件 2 也有 test
对象
在文件 3
require 'file1.php';require 'file2.php';
那么就会出现冲突
test1.php
namespace test\file1;class test { public $val = 'this is file 1';}
test2.php
namespace test\file2; # 也可以使用别的命名空间 ...\...\...\...class test { public $val = 'this is file 2';}
test.php 用于加载访问
require 'test1.php';require 'test2.php';$t1 = new test\file1\test();$t2 = new test\file2\test();echo $t1->val, '
'; // this is file1echo $t2->val, '
'; // this is file2use test\file1 as file1;use test\file2 as file2;$t3 = new file1\test();$t4 = new file2\test();echo $t3->val, '
'; // this is file1echo $t4->val, '
'; // this is file2
class_alias(class_name, new_classname)get_class (OBJ)$arr = get_declared_classes() # 返回已经定义了的类get_parent_class (OBJ)interface_exists (INTER) # 是否存在接口method_exists (OBJ, FUNC_NAME) # 是否存在某个方法
function __autoload ( $class_name ) { require $class_name . '.php' ;}$obj = new MyClass1 (); // 自动会调用上面的哪个函数, 参数名是 MyClass1$obj2 = new MyClass2 (); // 自动会调用上面的哪个函数, 参数名是 MyClass2
为什么函数里面的 require class 的生活周期可以到外层, 原因像下面的,
__autoload 被执行之后, 就相当于把函数里面的代码展示出来
function well() { function test() { echo 'inside ', '
'; }}well(); // 一定要先调用test(); // inside
__autoload 的实现spl_autoload_register —— 注册给定的函数作为 __autoload 的实现
function func() { require '...';}spl_autoload_register(func);