最近学习 ThinkPHP5,第一次看到 TestClass::instance() 就能创建 TestClass 实例的方法。感到很好奇,翻阅 ThinkPHP 的源码,大体理解了它的 设计思想,非常的先进。
代码:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
<?php
class
TestClass {
public
static
function
instance() {
return
new
self();
}
public
$data
= [];
public
function
__set(
$name
,
$val
) {
return
$this
->data[
$name
] =
$val
;
}
public
function
__get(
$name
) {
return
$this
->data[
$name
];
}
}
$app1
= TestClass::instance();
$app1
->key =
'Application 1'
;
echo
$app1
->key .
'<br />'
;
?>
|
为了方便调用,也模仿 ThinkPHP 写了一个助手函数
1
2
3
4
5
6
7
8
9
|
<?php
function
app() {
return
TestClass::instance();
}
$app2
= app();
$app2
->key =
'Application 2'
;
echo
$app2
->key .
'<br />'
;
?>
|
这样就简单的实现了 instance。
不过这种方法还有一个小问题,试想以下,调用100次,就需要创建100个实例,想想都觉得可怕。
给 Test 类 增加一个 静态属性,将创建的实例保存到这里。下次如果需要调用,则直接调用这个实例。
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
52
53
54
55
|
<?php
class
TestClass {
public
static
$instance
;
//用于缓存实例
public
$data
= [];
public
static
function
instance() {
//如果不存在实例,则返回实例
if
(
empty
(self::
$instance
)) {
self::
$instance
=
new
self();
}
return
self::
$instance
;
}
public
function
__set(
$name
,
$val
) {
return
$this
->data[
$name
] =
$val
;
}
public
function
__get(
$name
) {
return
$this
->data[
$name
];
}
}
function
app(
$option
= []) {
return
TestClass::instance(
$option
);
}
header(
'content-type:text/plain'
);
$result
= [];
$app1
= app();
$app1
->key =
"Application 1"
;
//修改 key 为 Application 1
$result
[
'app1'
] = [
'app1'
=>
$app1
->key,
//实例中 key 为 Application 1
];
// 创建 app2,因为 instance 已经存在实例,直接返回 缓存的实例
$app2
= app();
$result
[
'app2'
] = [
'setp1'
=> [
'app1'
=>
$app1
->key,
// Application 1
'app2'
=>
$app2
->key,
//因为直接调用的实例的缓存,所以 key 也是 Application 1
],
];
// 无论 app1,app2 都对在内存中 对应的同一个实例,无论通过谁修改,都能改变值
$app1
->key =
"Application 2"
;
$result
[
'app2'
][
'setp2'
] = [
'app1'
=>
$app1
->key,
// Application 2
'app2'
=>
$app2
->key,
// Application 2
];
print_r(
$result
);
?>
|
通过上边的实验,可以看到 无论调用多少次,都会使用同一个实例。这样就解决了效率低的问题。
到现在基本就满足大多数情况了,唯一的小缺陷,就是 可能 实例的 初始参数不同,这样没法灵活调用(常见的比如同一个程序调用两个数据库)。在 上边的 例子中稍作改造,以传入的参数为key,将不通的 实例缓存到数组中 就可以解决。
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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
<?php
class
TestClass {
public
static
$instance
= [];
//用于缓存实例数组
public
$data
= [];
public
function
__construct(
$opt
= []) {
$this
->data =
$opt
;
}
public
static
function
instance(
$option
= []) {
// 根据传入的参数 通过 serialize 转换为字符串,md5 后 作为数组的 key
$instance_id
= md5(serialize(
$option
));
//如果 不存在实例,则创建
if
(
empty
(self::
$instance
[
$instance_id
])) {
self::
$instance
[
$instance_id
] =
new
self(
$option
);
}
return
self::
$instance
[
$instance_id
];
}
public
function
__set(
$name
,
$val
) {
return
$this
->data[
$name
] =
$val
;
}
public
function
__get(
$name
) {
return
$this
->data[
$name
];
}
}
function
app(
$option
= []) {
return
TestClass::instance(
$option
);
}
header(
'content-type:text/plain'
);
$result
= [];
//传入 初始数据
$app1
= app([
'key'
=>
'123'
]);
$result
[
'init'
] =
$app1
->key;
// 使用 传入的数据,即:123
$app1
->key =
"app1"
;
$result
[
'app'
] =
$app1
->key;
// 现在值改为了 自定义的 app1了
print_r(
$result
);
$result
= [];
// 创建 app2,注意 初始参数不一样
$app2
= app();
// 因为初始参数不一样,所以还是创建新的实例
$app2
->key =
"app2"
;
$result
[
'app1'
] =
$app1
->key;
// app1
$result
[
'app2'
] =
$app2
->key;
// app2
print_r(
$result
);
$result
= [];
// 创建 app3,传入的参数 和 app1 一样,所以会直接返回 和app1相同 的 实例
$app3
= app([
'key'
=>
'123'
]);
$result
[
'log'
] = [
'app1'
=>
$app1
->key,
// app1
'app2'
=>
$app2
->key,
// app2
'app3'
=>
$app3
->key,
// app1
];
// 设置 app3 的key,会自动修改 app1 的值,因为他们两个是同一个实例
$app3
->key =
'app3'
;
$result
[
'app3_set'
] = [
'app1'
=>
$app1
->key,
// app3
'app2'
=>
$app2
->key,
// app2
'app3'
=>
$app3
->key,
// app3
];
// 同理,设置 app1 的key,app3 的 key 也会修改
$app1
->key =
'app1'
;
$result
[
'app1_set'
] = [
'app1'
=>
$app1
->key,
// app1
'app2'
=>
$app2
->key,
// app2
'app3'
=>
$app3
->key,
// app1
];
print_r(
$result
);
?>
|