Closure 闭包
代表一个匿名函数的类,我们所用到的匿名函数,都是Closure的一个实例,主要方法有:bind ,bindTo
- bindTo 他有两个参数
public Closure Closure::bindTo ( object $newthis [, mixed $newscope = 'static' ] )
$newthis是指需要绑定的对象,newscope是设置类的作用域 第一个参数可以设置null或者是一个对象的实例,主要取决于我们要在函数中操作的属性和方法是否是静态的,如果是静态的,那么第一个参数必须设置null,且第二个参数要设置一个类作用域
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
class A {
private $number = 'one';
}
$newfunc = function(){
$this->number = 'two';
echo $this->number;
}
$a = new A;
$func = $newfunc->bindTo($a,A::class);
$func();
//输出 'two'
//操作静态属性
class A {
private static $number = 'one';
}
$newfunc function(){
self::number = 'two';
echo self::number;
}
$func = $newfunc->bindTo(null,A::class);
$func();
//输出 'two'
- 静态方法 bind 他有三个参数
public static Closure Closure::bind ( Closure $closure , object $newthis [, mixed $newscope = 'static' ] )
$closure是指我们需要绑定的匿名函数,$newthis是指需要绑定的对象,newscope是设置类的作用域 bind其实和bindTo用法一致,只不过在于使用的方式不一样而已
1
2
3
4
5
6
7
8
9
10
11
12
13
class A {
private $number = 'one';
}
$a = new A;
$func = Closure::bind(function(){
$this->number = 'two';
echo $this->number;
},$a,A::class);
$func();
//输出 'two'
ArrayAccess 数组式访问
很多框架中,明明是数据格式是对象,却能以数组的形式访问,这就是用到了数组式访问
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
class A implements ArrayAccess{
public $name;
public function __construct(){
$this->name = 'one';
}
public function offsetSet($offset, $value) {
$this->$offset = $value;
}
public function offsetExists($offset) {
return isset($this->$offset);
}
public function offsetUnset($offset) {
unset($this->$offset);
}
public function offsetGet($offset) {
return $this->$offset;
}
}
$a = new A;
// 这里会调用offsetGet
echo $a['name'];
// 这里会调用offsetSet
$a['name'] = 'two';
echo $a['name'];
// 这里会调用offsetExists
var_dump(isset($a['name']));
// 这里会调用offsetUnset
unset($a['name']);
var_dump(isset($a['name']));
Traversable 遍历
检测一个类是否可以用foreach进行遍历,无法被单独实现的基本抽象接口
1
2
3
var_dump(new testClass instanceof Traversable);
//输出 false
Iterator 迭代器
主要功能是用来遍历一个对象,正常遍历对象只能获取pulic属性,不能获取protect域private属性.在不了解对象结果的情况下,Iterator可以帮助我们去获取他的所有的属性内容
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
class Test implements Iterator {
private $position = 0;
private $array = ['nine' , 'seven'];
// 该方法主要用户项目初始化
function rewind() {
echo __METHOD__ . PHP_EOL;
$this->position = 0;
}
//用来获取当前游标所对应的值
function current() {
echo __METHOD__ . PHP_EOL;
return $this->array[$this->position];
}
//获取当前游标
function key() {
echo __METHOD__ . PHP_EOL;
return $this->position;
}
//下移游标
function next() {
echo __METHOD__ . PHP_EOL;
++$this->position;
}
//判断是否还有值
function valid() {
echo __METHOD__ . PHP_EOL;
return isset($this->array[$this->position]);
}
}
$obj = new Test;
$obj->rewind();
while($obj->valid()){
echo $obj->current() . PHP_EOL;
$obj->next();
}
//输出
Test::rewind
Test::valid
Test::current
nine
Test::next
Test::valid
Test::current
seven
Test::next
Test::valid
IteratorAggregate 聚合式迭代器
IteratorAggregate接口继承了Traversable,主要用来创建外部迭代器的接口,只需要实现一个getIterator方法即可
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
class Test implements IteratorAggregate {
protected $name = 'one';
public $age = 18;
public function getIterator(){
return new ArrayIterator($this);
}
}
$obj = (new Test)->getIterator();
$obj->rewind();
while($obj->valid()){
echo $obj->current() . PHP_EOL;
$obj->next();
}
//输出
18
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
class Test implements IteratorAggregate {
private $_data;
public function __construct(){
$this->_data = ['one' , 'two'];
}
public function getIterator(){
return new ArrayIterator($this->_data);
}
}
$obj = (new Test)->getIterator();
$obj->rewind();
while($obj->valid()){
echo $obj->current() . PHP_EOL;
$obj->next();
}
//输出 one two
输出内容取决于给new ArrayIterator
注入的数组,ArrayIterator
继承于Iterator
,实现了Iterator
中的几个方法
Serializable 序列化
不论何时,只要有实例需要被序列化,serialize()
方法都将被调用。它将不会调用 __destruct()
或有其他影响,除非程序化地调用此方法当数据被反序列化时,类将被感知并且调用合适的 unserialize()
方法而不是调用 __construct()
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
class MyClass implements Serializable {
private $data;
public function __construct($data) {
$this->data = $data;
}
public function serialize() {
return serialize($this->data);
}
public function unserialize($data) {
$this->data = unserialize($data);
}
}
$a = new MyClass('hello , world');
var_dump($a->serialize());
//输出
string(21) "s:13:"hello , world";"
Generator 生成器
Generator
实现了Iterator
,但是他无法被继承,同时也生成实例,由于实现了Iterator
,它便有了和Iterator
相同的功能,Generator
的语法主要来自于关键字yield,会消耗更少的内存
1
2
3
4
5
6
7
8
9
10
11
12
13
14
$start_time = microtime(true);
function xrange($num = 100000){
for($i = 0 ; $i < $num ; ++ $i){
yield $i;
}
}
$generator = xrange();
foreach ($generator as $key => $value) {
echo $key . '=' . $value . PHP_EOL;
}
echo 'memory:' . memory_get_usage() . ' time:' . (microtime(true) - $start_time) . PHP_EOL;
//输出memory:229056 time:0.25725412368774
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
$start_time = microtime(true);
function xrange2($num = 100000){
$arr = [];
for ($i=0; $i <$num ; ++$i) {
array_push($arr , $i);
}
return $arr;
}
$arr = xrange2();
foreach ($arr as $key => $value) {
# code...
}
echo 'memory:' . memory_get_usage() . ' time:' . (microtime(true) - $start_time);
//输出
memory:14877528 time:0.039144992828369