说一下这个Laravel5.3之函数处理(Function Handling)

说明:Laravel中经常使用PHP的Function Handling来设计代码,本文主要学习PHP的Function Handling特性,来提高写代码时的设计质量。PHP提供了一些函数处理操作的内置函数,主要有:

  1. call_user_func_array( )
  2. call_user_func( )
  3. func_get_arg( )
  4. func_get_args( )
  5. func_num_args( )
  6. 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中也大量使用了这个技术来巧妙设计代码。下次遇到好的技术在分享,到时见。

正文完