- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法分析由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
本文实例讲述了thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法。分享给大家供大家参考,具体如下:
tp5.1中引入了容器(Container)和门面(Facade)这两个新的类 。
官方文档已经给出了定义:
容器(Container)实现类的统一管理,确保对象实例的唯一性.
门面(Facade)为容器(Container)中的类提供了一个静态调用接口,相比于传统的静态方法调用, 带来了更好的可测试性和扩展性,你可以为任何的非静态类库定义一个facade类.
深入源码,我们来看看它到底是如何实现的:
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
|
// 在框架目录下的base.php文件
// 注册核心类到容器
Container::getInstance()->bind([
'app'
=> App::
class
,
'build'
=> Build::
class
,
'cache'
=> Cache::
class
,
'config'
=> Config::
class
,
...
]);
// 注册核心类的静态代理
Facade::bind([
facade\App::
class
=> App::
class
,
facade\Build::
class
=> Build::
class
,
facade\Cache::
class
=> Cache::
class
,
facade\Config::
class
=> Config::
class
,
...
]);
// 注册类库别名
Loader::addClassAlias([
'App'
=> facade\App::
class
,
'Build'
=> facade\Build::
class
,
'Cache'
=> facade\Cache::
class
,
'Config'
=> facade\Config::
class
,
...
]);
|
容器实现:
这里,框架已经帮我们绑定了系统常用类到容器中,在之后使用时,只需要调用助手函数 app()进行容器中的类解析调用,对于已经绑定的类标识,会自动快速实例化.
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
|
// 实例化缓存类
app(
'cache'
);
// app('cache', ['file']); 参数化调用
// 相当于执行了
Container::get(
'cache'
);
// 查看源码,Container调用的其实是make方法,在该方法里调用反射等实现类的实例化,过程如下:
public
function
make(
$abstract
,
$vars
= [],
$newInstance
= false)
{
if
(true ===
$vars
) {
// 总是创建新的实例化对象
$newInstance
= true;
$vars
= [];
}
if
(isset(
$this
->instances[
$abstract
]) && !
$newInstance
) {
$object
=
$this
->instances[
$abstract
];
}
else
{
if
(isset(
$this
->bind[
$abstract
])) {
$concrete
=
$this
->bind[
$abstract
];
// 闭包实现
if
(
$concrete
instanceof
\Closure) {
$object
=
$this
->invokeFunction(
$concrete
,
$vars
);
}
else
{
$object
=
$this
->make(
$concrete
,
$vars
,
$newInstance
);
}
}
else
{
// 反射实现
$object
=
$this
->invokeClass(
$abstract
,
$vars
);
}
if
(!
$newInstance
) {
$this
->instances[
$abstract
] =
$object
;
}
}
return
$object
;
}
/**
* 调用反射执行类的实例化 支持依赖注入
* @access public
* @param string $class 类名
* @param array $vars 变量
* @return mixed
*/
public
function
invokeClass(
$class
,
$vars
= [])
{
$reflect
=
new
\ReflectionClass(
$class
);
$constructor
=
$reflect
->getConstructor();
if
(
$constructor
) {
$args
=
$this
->bindParams(
$constructor
,
$vars
);
}
else
{
$args
= [];
}
return
$reflect
->newInstanceArgs(
$args
);
}
/**
* 执行函数或者闭包方法 支持参数调用
* @access public
* @param string|array|\Closure $function 函数或者闭包
* @param array $vars 变量
* @return mixed
*/
public
function
invokeFunction(
$function
,
$vars
= [])
{
$reflect
=
new
\ReflectionFunction(
$function
);
$args
=
$this
->bindParams(
$reflect
,
$vars
);
return
$reflect
->invokeArgs(
$args
);
}
|
简而言之,容器内部是通过反射类或闭包等来实现类的实例化.
门面实现:
以一个例子来分析:
1
|
facade\Config::get(
'app_debug'
);
|
我们来分析一下它的实现方式:
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
|
// thinkphp\library\facade\Config 类
namespace
think\facade;
use
think\Facade;
class
Config
extends
Facade
{
}
// 从源代码上看 Config本身没有任何方法,它继承了Facade的方法,但Facade并没有get这个静态方法
// 此时,系统自动触发了魔术方法:__callStatic(),Facade重写了此方法:
public
static
function
__callStatic(
$method
,
$params
)
{
return
call_user_func_array([
static
::createFacade(),
$method
],
$params
);
}
// 可见,最后调用的是用户自定义函数:call_user_func_array([实例, 方法], 参数),为了获得Config实例,Facade又定义了一个获取对象的方法:
/**
* 创建Facade实例
* @static
* @access protected
* @param string $class 类名或标识
* @param array $args 变量
* @param bool $newInstance 是否每次创建新的实例
* @return object
*/
protected
static
function
createFacade(
$class
=
''
,
$args
= [],
$newInstance
= false)
{
$class
=
$class
?:
static
::
class
;
$facadeClass
=
static
::getFacadeClass();
if
(
$facadeClass
) {
$class
=
$facadeClass
;
}
elseif
(isset(self::
$bind
[
$class
])) {
$class
= self::
$bind
[
$class
];
}
if
(
static
::
$alwaysNewInstance
) {
$newInstance
= true;
}
return
Container::getInstance()->make(
$class
,
$args
,
$newInstance
);
}
// 其内部是通过容器来实例化对象
// 因为在base.php中已经将 think\Config 类绑定到 config 这个标识
Container::getInstance()->bind([
'config'
=> Config::
class
])
// 在 createFacade 方法中,获取类的名称:$class = $class ?: static::class; 即得到 config 这个标识
// 在容器的make方法中,根据config标识,找到绑定的 think\Config 类,并调用其动态方法 get。
facade\Config::get(
'app_debug'
);
// 最后调用的是:
(
new
think\Config())->get(
'app_debug'
);
|
简而言之,门面的实现是通过PHP的魔术方法 __callStatic,再配合容器来实现动态类的静态化调用.
希望本文所述对大家基于ThinkPHP框架的PHP程序设计有所帮助.
原文链接:https://www.cnblogs.com/cqingt/p/8243294.html 。
最后此篇关于thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法分析的文章就讲到这里了,如果你想了解更多关于thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法分析的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
我刚刚继承了一个旧的 PostgreSQL 安装,需要进行一些诊断以找出该数据库运行缓慢的原因。在 MS SQL 上,您可以使用 Profiler 等工具来查看正在运行的查询,然后查看它们的执行计划。
将目标从Analytics(分析)导入到AdWords中,然后在Analytics(分析)中更改目标条件时,是否可以通过更改将目标“重新导入”到AdWords,还是可以自动选择? 最佳答案 更改目标值
我正在使用google analytics api来获取数据。我正在获取数据,但我想验证两个参数,它们在特定日期范围内始终为0。我正在获取['ga:transactions']和['ga:goalCo
我使用Google API从Google Analytics(分析)获取数据,但指标与Google Analytics(分析)的网络界面不同。 即:我在2015年3月1日获得数据-它返回综合浏览量79
我在我的Web应用程序中使用sammy.js进行剔除。我正在尝试向其中添加Google Analytics(分析)。我很快找到了following plugin来实现页面跟踪。 我按照步骤操作,页面如
当使用 Xcode 分析 (product>analyze) 时,有没有办法忽略给定文件中的任何错误? 例如编译指示之类的? 我们只想忽略第三方代码的任何警告,这样当我们的代码出现问题时,它对我们
目录 EFK 1. 日志系统 2. 部署ElasticSearch 2.1 创建handless服务 2.2 创建s
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 7年前关闭。 Improve thi
GCC/G++ 是否有可用于输出分析的选项? 能够比较以前的代码与新代码之间的差异(大小、类/结构的大小)将很有用。然后可以将它们与之前的输出进行比较以进行比较,这对于许多目的都是有用的。 如果没有此
我正在浏览 LYAH,并一直在研究处理列表时列表理解与映射/过滤器的使用。我已经分析了以下两个函数,并包含了教授的输出。如果我正确地阅读了教授的内容,我会说 FiltB 的运行速度比 FiltA 慢很
在 MySQL 中可以使用 SET profiling = 1; 设置分析 查询 SHOW PROFILES; 显示每个查询所用的时间。我想知道这个时间是只包括服务器的执行时间还是还包括将结果发送到前
我用 Python 编写了几个用于生成阶乘的模块,我想测试运行时间。我找到了一个分析示例 here我使用该模板来分析我的模块: import profile #fact def main():
前几天读了下mysqld_safe脚本,个人感觉还是收获蛮大的,其中细致的交代了MySQL数据库的启动流程,包括查找MySQL相关目录,解析配置文件以及最后如何调用mysqld程序来启动实例等,有着
上一篇:《人工智能大语言模型起源篇,低秩微调(LoRA)》 (14)Rae 和同事(包括78位合著者!)于2022年发表的《Scaling Language Models: Methods, A
1 内网基础 内网/局域网(Local Area Network,LAN),是指在某一区域内有多台计算机互联而成的计算机组,组网范围通常在数千米以内。在局域网中,可以实现文件管理、应用软件共享、打印机
1 内网基础 内网/局域网(Local Area Network,LAN),是指在某一区域内有多台计算机互联而成的计算机组,组网范围通常在数千米以内。在局域网中,可以实现文件管理、应用软件共享、打印机
我有四列形式的数据。前三列代表时间,value1,value 2。第四列是二进制,全为 0 或 1。当第四列中对应的二进制值为0时,有没有办法告诉excel删除时间、值1和值2?我知道这在 C++ 或
我正在运行一个进行长时间计算的 Haskell 程序。经过一些分析和跟踪后,我注意到以下内容: $ /usr/bin/time -v ./hl test.hl 9000045000050000 Com
我有一个缓慢的 asp.net 程序正在运行。我想分析生产服务器以查看发生了什么,但我不想显着降低生产服务器的速度。 一般而言,配置生产盒或仅本地开发盒是标准做法吗?另外,您建议使用哪些程序来实现这一
我目前正在尝试分析 Haskell 服务器。服务器永远运行,所以我只想要一个固定时间的分析报告。我尝试只运行该程序 3 分钟,然后礼貌地要求它终止,但不知何故,haskell 分析器不遵守术语信号,并
我是一名优秀的程序员,十分优秀!