说明:Laravel中经常使用PHP的Function Handling
来设计代码,本文主要学习PHP的Function Handling
特性,来提高写代码时的设计质量。PHP提供了一些函数处理操作的内置函数,主要有:
-
call_user_func_array( )
-
call_user_func( )
-
func_get_arg( )
-
func_get_args( )
-
func_num_args( )
-
function_exists( )
开发环境:Laravel5.3 + PHP7
Function Handling
call_user_func_array()/call_user_func()
call_user_func_array()是调用回调函数,并把一个数组作为参数传进去作为回调函数的参数;call_user_func()也是调用回调函数,区别是并没有要求把数组作为参数传进回调函数做参数。在Laravel中大量使用这两个内置函数来设计代码,比如\Illuminate\Foundation\Application::fireAppCallbacks()的源码:
/** * Call the booting callbacks for the application. * * @param array $callbacks * @return void */
protected function fireAppCallbacks(array $callbacks)
{
foreach ($callbacks as $callback) {
call_user_func($callback, $this); //执行回调函数,并把Application对象作为参数传进去
}
}
call_user_func()和call_user_func_array()可以说是PHP设计好代码的神器,不得不熟悉,这里给下它的PHPUnit测试看看如何使用,爆绿灯:
<?php
namespace MyRightCapital\Container\Tests;
class FunctionHandling extends \PHPUnit_Framework_TestCase
{
public function testCallUserFunc()
{
// Arrange
$provider = new Provider();
$app = new Application($provider);
// Actual
$actual = call_user_func('MyRightCapital\Container\Tests\callUserFunc', $app);
// Assert
$this->assertSame('This is a service provider.', $actual);
}
public function testCallUserFuncArray()
{
// Arrange
$provider = new Provider();
$app = new Application($provider);
// Actual
$actual = call_user_func_array('MyRightCapital\Container\Tests\callUserFunc', [$app]);
// Assert
$this->assertSame('This is a service provider.', $actual);
}
}
function callUserFunc($app)
{
return $app->register();
}
class Application
{
private $provider;
public function __construct($provider)
{
$this->provider = $provider;
}
public function register()
{
return $this->provider->register();
}
}
class Provider
{
public function register()
{
return 'This is a service provider.';
}
}
call_user_func_array()和call_user_func()真是个非常用的函数,值得在设计自己的代码里使用。
func_get_arg()/func_get_args()/func_num_args()
func_get_arg()是从函数的参数列表读取某个指定的参数,func_get_args()是读取函数的整个参数列表作为数组返回,func_num_args()是读取函数的参数的个数。Laravel中的IlluminateFoundationApplication::environment()
使用了这三个函数来设计代码,很巧妙:
/** * Get or check the current application environment. * * @return string|bool */
public function environment()
{
// 如果传入了参数
if (func_num_args() > 0) {
// 如果第一个参数是数组形式就把该数组赋值给$patterns;如果不是就把所有参数作为一个数组赋值给$patterns
$patterns = is_array(func_get_arg(0)) ? func_get_arg(0) : func_get_args();
foreach ($patterns as $pattern) {
if (Str::is($pattern, $this['env'])) {
return true;
}
}
return false;
}
return $this['env'];
}
看environment()源码可知道environment()是可以传入参数的,如果不传入参数就返回$this['env']
的值即Laravel中的环境变量APP_ENV值
,如App::environment()
即为读取Laravel当前运行环境变量值;如果传入参数则判断该值是否与环境变量值相等,如App::environment('production','staging', 'development')
即判断当前Laravel运行环境是否是'production','staging', 'development'
中的一种。很巧妙的设计。
这里写个PHPUnit测试下,爆绿灯:
class FunctionHandling extends \PHPUnit_Framework_TestCase
{
public function testFuncArgs()
{
// Arrange
$provider = new Provider();
$app = new Application($provider);
// Actual
$arg_number0 = $app->testFuncArg();
$arg_number1 = $app->testFuncArg('Laravel');
$arg_number2 = $app->testFuncArg(['Laravel', 'PHP']);
// Assert
$this->assertSame(0, $arg_number0);
$this->assertSame(1, $arg_number1);
$this->assertSame(2, $arg_number2);
}
}
class Application
{
private $provider;
public function __construct($provider)
{
$this->provider = $provider;
}
public function register()
{
return $this->provider->register();
}
public function testFuncArg()
{
if (func_num_args() > 0) {
$patterns = is_array(func_get_arg(0)) ? func_get_arg(0) :func_get_args();
return count($patterns);
}
return 0;
}
}
function_exists()
function_exists()判断指定函数是否已经定义,这个函数在Laravel中大量使用,尤其是造辅助函数时使用,参考Illuminate/Foundation/helpers.php,Illuminate/Support/helpers.php。这里做个PHPUnit测试,爆绿灯:
class FunctionHandling extends \PHPUnit_Framework_TestCase
{
public function testFunctionExists()
{
// Arrange
$expected = 'Container';
// Actual
$actual = functionExists('Container');
// Assert
$this->assertSame($expected, $actual);
}
}
if (!function_exists('functionExists')) {
function functionExists($container)
{
return $container;
}
}
总结:本文主要学习了PHP的Function Handling,这个技术可以用来提高自己的代码设计能力,同时Laravel中也大量使用了这个技术来巧妙设计代码。下次遇到好的技术在分享,到时见。