>  기사  >  백엔드 개발  >  PHP 컨테이너 Pimple의 실행 프로세스 분석

PHP 컨테이너 Pimple의 실행 프로세스 분석

不言
不言원래의
2018-07-06 13:45:173568검색

이 글은 주로 PHP 컨테이너 Pimple의 실행 프로세스에 대한 분석을 소개합니다. 이제는 모든 사람과 공유합니다. 도움이 필요한 친구들이 참고할 수 있습니다.

지식 포인트를 알아야 합니다

Closure

Closure 익명 기능은 PHP5.3.0에서 도입되었습니다.

클로저란 생성 시 주변 상태를 캡슐화하는 기능을 말합니다. 클로저가 위치한 환경이 더 이상 존재하지 않더라도 클로저에 캡슐화된 상태는 여전히 존재합니다.

이론적으로 클로저와 익명 함수는 다른 개념입니다. 그러나 PHP는 이를 동일한 개념으로 취급합니다.
사실 클로저와 익명 함수는 함수로 위장한 객체입니다. 이는 Closure 클래스의 인스턴스입니다.

문자열 및 정수와 같은 클로저는 최고 수준의 값 유형입니다.

클로저 만들기:

<?php
$closure = function ($name) {
    return &#39;Hello &#39; . $name;
};
echo $closure(&#39;nesfo&#39;);//Hello nesfo
var_dump(method_exists($closure, &#39;__invoke&#39;));//true
$closure 변수를 호출할 수 있는 이유는 이 변수의 값이 클로저이고 클로저 개체가 __invoke() 코드를 구현하기 때문입니다. >마법의 방법. 변수 이름 뒤에 <code>()가 있으면 PHP는 __invoke() 메서드를 찾아 호출합니다. $closure变量,是因为这个变量的值是一个闭包,而且闭包对象实现了__invoke()魔术方法。只要变量名后有(),PHP就会查找并调用__invoke()方法。

通常会把PHP闭包当作函数的回调使用。

array_map(), preg_replace_callback()方法都会用到回调函数,这是使用闭包的最佳时机!

举个例子:

<?php
$numbersPlusOne = array_map(function ($number) {
    return $number + 1;
}, [1, 2, 3]);
print_r($numbersPlusOne);

得到结果:

[2, 3, 4]

在闭包出现之前,只能单独创建具名函数,然后使用名称引用那个函数。这么做,代码执行会稍微慢点,而且把回调的实现和使用场景隔离了。

<?php
function incrementNum ($number) {
    return $number + 1;
}

$numbersPlusOne = array_map(&#39;incrementNum&#39;, [1, 2, 3]);
print_r($numbersPlusOne);

SPL

ArrayAccess

实现ArrayAccess接口,可以使得object像array那样操作。ArrayAccess接口包含四个必须实现的方法:

interface ArrayAccess {
    //检查一个偏移位置是否存在 
    public mixed offsetExists ( mixed $offset  );
    
    //获取一个偏移位置的值 
    public mixed offsetGet( mixed $offset  );
    
    //设置一个偏移位置的值 
    public mixed offsetSet ( mixed $offset  );
    
    //复位一个偏移位置的值 
    public mixed offsetUnset  ( mixed $offset  );
}

SplObjectStorage

SplObjectStorage类实现了以对象为键的映射(map)或对象的集合(如果忽略作为键的对象所对应的数据)这种数据结构。这个类的实例很像一个数组,但是它所存放的对象都是唯一。该类的另一个特点是,可以直接从中删除指定的对象,而不需要遍历或搜索整个集合。

::class语法

因为 ::class 表示是字符串。用 ::class 的好处在于 IDE 里面可以直接改名一个 class,然后 IDE 自动处理相关引用。
同时,PHP 执行相关代码时,是不会先加载相关 class 的。

同理,代码自动化检查 inspect 也可以正确识别 class。

Pimple容器流程浅析

Pimpl是php社区中比较流行的容器。代码不是很多,详见https://github.com/silexphp/P... 。

我们的应用可以基于Pimple开发:

namespace EasyWeChat\Foundation;

use Pimple\Container;

class Application extends Container
{
    /**
     * Service Providers.
     *
     * @var array
     */
    protected $providers = [
        ServiceProviders\ServerServiceProvider::class,
        ServiceProviders\UserServiceProvider::class
    ];

    /**
     * Application constructor.
     *
     * @param array $config
     */
    public function __construct($config)
    {
        parent::__construct();

        $this[&#39;config&#39;] = function () use ($config) {
            return new Config($config);
        };

        if ($this[&#39;config&#39;][&#39;debug&#39;]) {
            error_reporting(E_ALL);
        }

        $this->registerProviders();
    }

    /**
     * Add a provider.
     *
     * @param string $provider
     *
     * @return Application
     */
    public function addProvider($provider)
    {
        array_push($this->providers, $provider);

        return $this;
    }

    /**
     * Set providers.
     *
     * @param array $providers
     */
    public function setProviders(array $providers)
    {
        $this->providers = [];

        foreach ($providers as $provider) {
            $this->addProvider($provider);
        }
    }

    /**
     * Return all providers.
     *
     * @return array
     */
    public function getProviders()
    {
        return $this->providers;
    }

    /**
     * Magic get access.
     *
     * @param string $id
     *
     * @return mixed
     */
    public function __get($id)
    {
        return $this->offsetGet($id);
    }

    /**
     * Magic set access.
     *
     * @param string $id
     * @param mixed  $value
     */
    public function __set($id, $value)
    {
        $this->offsetSet($id, $value);
    }
}

如何使用我们的应用:

$app = new Application([]);
$user = $app->user;

之后我们就可以使用$user对象的方法了。我们发现其实并没有$this->user这个属性,但是可以直接使用。主要是这两个方法起的作用:

public function offsetSet($id, $value){}
public function offsetGet($id){}

下面我们将解释在执行这两句代码,Pimple做了什么。但在解释这个之前,我们先看看容器的一些核心概念。

服务提供者

服务提供者是连接容器与具体功能实现类的桥梁。服务提供者需要实现接口ServiceProviderInterface:

namespace Pimple;

/**
 * Pimple service provider interface.
 *
 * @author  Fabien Potencier
 * @author  Dominik Zogg
 */
interface ServiceProviderInterface
{
    /**
     * Registers services on the given container.
     *
     * This method should only be used to configure services and parameters.
     * It should not get services.
     *
     * @param Container $pimple A container instance
     */
    public function register(Container $pimple);
}

所有服务提供者必须实现接口register方法。

我们的应用里默认有2个服务提供者:

protected $providers = [
    ServiceProviders\ServerServiceProvider::class,
    ServiceProviders\UserServiceProvider::class
];

以UserServiceProvider为例,我们看其代码实现:

namespace EasyWeChat\Foundation\ServiceProviders;

use EasyWeChat\User\User;
use Pimple\Container;
use Pimple\ServiceProviderInterface;

/**
 * Class UserServiceProvider.
 */
class UserServiceProvider implements ServiceProviderInterface
{
    /**
     * Registers services on the given container.
     *
     * This method should only be used to configure services and parameters.
     * It should not get services.
     *
     * @param Container $pimple A container instance
     */
    public function register(Container $pimple)
    {
        $pimple[&#39;user&#39;] = function ($pimple) {
            return new User($pimple[&#39;access_token&#39;]);
        };
    }
}

我们看到,该服务提供者的注册方法会给容器增加属性user,但是返回的不是对象,而是一个闭包。这个后面我再做讲解。

服务注册

我们在Application里构造函数里使用$this->registerProviders();对所有服务提供者进行了注册:

private function registerProviders()
{
    foreach ($this->providers as $provider) {
        $this->register(new $provider());
    }
}

仔细看,我们发现这里实例化了服务提供者,并调用了容器Pimple的register方法:

public function register(ServiceProviderInterface $provider, array $values = array())
{
    $provider->register($this);

    foreach ($values as $key => $value) {
        $this[$key] = $value;
    }

    return $this;
}

而这里调用了服务提供者的register方法,也就是我们在上一节中提到的:注册方法给容器增加了属性user,但返回的不是对象,而是一个闭包。

当我们给容器Pimple添加属性user的同时,会调用offsetSet($id, $value)方法:给容器Pimple的属性valueskeys分别赋值:

$this->values[$id] = $value;
$this->keys[$id] = true;

到这里,我们还没有实例化真正提供实际功能的类EasyWeChatUserUsr。但已经完成了服务提供者的注册工作。

当我们运行到这里:

$user = $app->user;

会调用offsetGet($id)并进行实例化真正的类:

$raw = $this->values[$id];
$val = $this->values[$id] = $raw($this);
$this->raw[$id] = $raw;

$this->frozen[$id] = true;

return $val;

$raw获取的是闭包:

$pimple[&#39;user&#39;] = function ($pimple) {
    return new User($pimple[&#39;access_token&#39;]);
};

$raw($this)返回的是实例化的对象User。也就是说只有实际调用才会去实例化具体的类。后面我们就可以通过$this['user']或者$this->user

보통 PHP 클로저는 함수의 콜백으로 사용됩니다.

array_map(), preg_replace_callback() 메소드는 모두 콜백 함수를 사용합니다. 이때가 클로저를 사용하기에 가장 좋은 시기입니다! 🎜🎜예: 🎜rrreee🎜결과 가져오기: 🎜rrreee🎜클로저 전에는 이름이 지정된 함수만 개별적으로 생성한 다음 해당 함수를 이름으로 참조할 수 있었습니다. 이렇게 하면 코드 실행이 약간 느려지고 콜백 구현이 사용 시나리오에서 격리됩니다. 🎜rrreee🎜SPL🎜

ArrayAccess

🎜객체가 배열처럼 작동할 수 있도록 하는 ArrayAccess 인터페이스를 구현합니다. ArrayAccess 인터페이스에는 구현되어야 하는 네 가지 메소드가 포함되어 있습니다: 🎜rrreee

SplObjectStorage

🎜SplObjectStorage 클래스는 객체를 키로 사용하거나 객체 컬렉션을 사용하여 맵을 구현합니다(키인 객체에 해당하는 데이터가 무시되는 경우). 이 데이터 구조. 이 클래스의 인스턴스는 배열과 매우 유사하지만 저장되는 개체는 모두 고유합니다. 이 클래스의 또 다른 특징은 전체 컬렉션을 탐색하거나 검색하지 않고도 지정된 개체를 클래스에서 직접 제거할 수 있다는 것입니다. 🎜🎜::class 구문🎜🎜::class는 문자열을 나타내기 때문입니다. ::class를 사용하면 IDE에서 클래스 이름을 직접 바꿀 수 있고 그러면 IDE가 관련 참조를 자동으로 처리한다는 장점이 있습니다. 🎜동시에 PHP는 관련 코드를 실행할 때 관련 클래스를 먼저 로드하지 않습니다. 🎜🎜마찬가지로 자동화된 코드 검사 검사도 클래스를 정확하게 식별할 수 있습니다. 🎜🎜Pimple 컨테이너 프로세스에 대한 간략한 분석🎜🎜Pimpl은 PHP 커뮤니티에서 인기 있는 컨테이너입니다. 코드가 많지 않습니다. 자세한 내용은 https://github.com/silexphp/P...를 참조하세요. 🎜🎜우리 애플리케이션은 Pimple을 기반으로 개발될 수 있습니다: 🎜rrreee🎜우리 애플리케이션 사용 방법: 🎜rrreee🎜그런 다음 $user 개체의 메서드를 사용할 수 있습니다. $this->user 속성은 없지만 직접 사용할 수 있는 것으로 확인되었습니다. 주로 이 두 가지 메소드의 역할: 🎜rrreee🎜 아래에서는 이 두 줄의 코드를 실행할 때 Pimple이 수행하는 작업에 대해 설명합니다. 하지만 이를 설명하기 전에 컨테이너의 몇 가지 핵심 개념을 살펴보겠습니다. 🎜🎜서비스 제공자🎜🎜서비스 제공자는 컨테이너와 특정 기능 구현 클래스를 연결하는 다리입니다. 서비스 제공자는 ServiceProviderInterface 인터페이스를 구현해야 합니다.🎜rrreee🎜모든 서비스 제공자는 인터페이스 register 메소드를 구현해야 합니다. 🎜🎜우리 애플리케이션에는 기본적으로 2개의 서비스 공급자가 있습니다. 🎜rrreee🎜 UserServiceProvider를 예로 들어 코드 구현을 살펴보겠습니다. 🎜rrreee🎜이 서비스 공급자의 등록 방법은 컨테이너에 속성을 추가하는 것을 볼 수 있습니다. > user이지만 반환되는 것은 객체가 아니라 클로저입니다. 이에 대해서는 나중에 설명하겠습니다. 🎜🎜서비스 등록🎜🎜모든 서비스 제공자를 등록하기 위해 애플리케이션 생성자에서 $this->registerProviders();를 사용합니다. 🎜rrreee🎜자세히 살펴보면 여기에 인스턴스화가 있음을 알 수 있습니다. 서비스 제공자가 호출되고 컨테이너 Pimple의 register 메소드가 호출됩니다. 🎜rrreee🎜여기서 서비스 제공자의 register 메소드가 호출됩니다. 이전 섹션에서 언급됨: 등록 방법은 user 속성을 ​​컨테이너에 추가하지만 객체가 아닌 클로저를 반환합니다. 🎜🎜Pimple 컨테이너에 user 속성을 ​​추가하면 values 속성에 대해 <code>offsetSet($id, $value) 메서드가 호출됩니다. 컨테이너의 Pimple 코드>, keys에는 각각 값이 할당됩니다. 🎜rrreee🎜지금까지는 실제로 실제 기능을 제공하는 EasyWeChatUserUsr 클래스를 인스턴스화하지 않았습니다. 다만, 서비스 제공자 등록이 완료되었습니다. 🎜🎜여기에서 실행하면: 🎜rrreee🎜는 offsetGet($id)를 호출하고 실제 클래스를 인스턴스화합니다. 🎜rrreee🎜$raw가 클로저를 가져옵니다. 🎜rrreee🎜$raw($this)는 인스턴스화된 개체 User를 반환합니다. 즉, 실제 호출만이 특정 클래스를 인스턴스화합니다. 나중에 $this['user'] 또는 $this->user를 통해 User 클래스의 메서드를 호출할 수 있습니다. 🎜🎜물론, Pimple에는 심층적으로 연구할 가치가 있는 많은 기능이 있으므로 여기서는 자세한 설명을 하지 않겠습니다. 🎜

이상은 이 글의 전체 내용입니다. 모든 분들의 학습에 도움이 되었으면 좋겠습니다. 더 많은 관련 내용은 PHP 중국어 홈페이지를 주목해주세요!

관련 권장 사항:

워드프레스가 wp_head() 함수를 사용하는 방법

#🎜🎜 #PHP 변수 범위, 전역, 정적 및 기타 키워드

위 내용은 PHP 컨테이너 Pimple의 실행 프로세스 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.