在学习 Laravel 源码前,有几个 PHP 的基础知识是必须熟悉的。基础知识熟悉之后,框架再复杂,也能慢慢地理解和上手。
Composer
Composer 是 PHP 的一个包依赖管理工具。你可以在自己的项目中声明所依赖的外部 Package ,Composer 会帮你安装这些依赖的包文件。比如你的项目准备使用
monolog/monolog 来记录日志,你只需要:
- 在命令行执行
composer require monolog/monolog
,Composer 会自动将这个包及所需的依赖文件下载到当前目录的 vendor 文件夹下。
monolog 目录.png - 在你的项目中引入 Composer 的自动加载文件。
require 'vendor/autoload.php';
然后就能使用 monolog 类了。
?php
use Monolog\Logger;
use Monolog\Handler\StreamHandler;
// create a log channel
$log = new Logger('name');
$log->pushHandler(new StreamHandler('path/to/your.log', Logger::WARNING));
Composer 的安装步骤可以参考官方的文档。
基本使用
- 常用 Composer 命令。
命令 | 说明 |
---|---|
composer install | 首先会根据 composer.lock 文件下载依赖;如果 composer.lock 文件没有,才会根据 composer.json 下载配置的依赖(并自动创建 composer.lock 文件)。下载后的依赖包会自动放在当前路径的 vendor 目录下 |
composer update | 根据你的 composer.json 文件更新依赖包(并自动更新 composer.lock 文件)。还可通过 composer update packageName 对单个包进行更新 |
composer dump-autoload | 更新 autoloader 自动加载。比如你添加了新的目录,想让他自动加载 |
composer validate | 检测你的 composer.json 文件是否有效 |
- 自动加载
通过 composer.json 安装的依赖包,引入vendor/autoload.php
文件后,系统会自动加载,非packagist.org
中的包,手动更新 composer.json 文件。
这里有4种方法引入其他的自动加载:
"autoload": {
// 目录中的文件会在 install/update 过程中生成,并存储到 vendor/composer/autoload_classmap.php 文件中
// 支持自定义加载的不遵循 PSR-0/4 规范的类库
"classmap": [
"database"
],
// PSR-4 规范的类库。 键为命名空间,值为对应的目录
"psr-4": {
"App\\": "app/"
},
// PSR-0 规范的类库
"psr-0": {
"Monolog\\": "src/"
},
// 每次请求时都会载入某些文件,通常作为通用函数库
"files": [
"app/helper.php"
]
},
PSR-4 规范参考,PSR-0 规范已经被废弃。
反射
通俗地说 反射 就是通过字符串 来获取对应类、类方法、类属性、类中方法和属性的注释的功能。比如系统中存在下面一个类:
<?php
class person {
private $age;
public function __construct() {
}
public function getAge() {
return $this->age;
}
public function setAge($age) {
$this->age = $age;
}
}
如果知道了类名 person , 要在框架中实例化, 最简单的方法是
$name = 'person';
$class = new $name();
但是如果 person 的构造方法中有参数呢,下面是通过反射获取一个类的实例简化的方法
function build($class) {
// 通过类名获取反射类
$reflector = new ReflectionClass($class);
// 获取类的构造函数
$constructor = $reflector->getConstructor();
// 获取构造函数的参数
$dependencies = $constructor->getParameters();
$instances = [];
// 获取参数对象和默认值
foreach ($dependencies as $dependency) {
// 获取参数的默认值
if ($dependency->isDefaultValueAvailable()) {
$instances[] = $dependency->getDefaultValue();
continue;
}
// 获取参数的对象
$instances[] = build($dependency->getClass()->name);
}
// 获取反射类的实例
return $reflector->newInstanceArgs($instances);
}
Laravel 框架的依赖注入以是使用 PHP 的反射来实例化类和构造方法中的对象的。
匿名函数
也叫闭包函数,是一个没有名字的函数。匿名函数默认是 Closure 类型的对象。
<?php
$add = function ($a, $b) {
return $a + $b;
};
echo ($add instanceof Closure) ? 'true':'false';
上面 $add 是一个匿名函数的变量,结果会打印出 true
。
在 Laravel 中匿名函数主要用于绑定类、注册服务等。
// 绑定闭包到一个抽象类
function singleton($abstract, $concrete = null) {
// ...
// 如果 $concrete 不是一个闭包,则将它构造成一个闭包
if (! $concrete instanceof Closure) {
$concrete = $this->getClosure($abstract, $concrete);
}
// 将闭包存入数组中
$this->bindings[$abstract] = compact('concrete', 'shared');
// ...
}
// 获取抽象类的实例
function resolve($abstract) {
// 获取绑定的内容
$concrete = $this->bindings[$abstract]['concrete'];
// 如果是闭包,则直接实例化
if ($concrete instanceof Closure) {
return $concrete($this, $this->getLastParameterOverride());
}
}
通过匿名函数,在绑定时可以不用实例化类,然后在需要用到类的时候再解析类,避免了不需要的实例化。
网友评论