찾다
백엔드 개발PHP 튜토리얼PHP 컨테이너 Pimple의 실행 프로세스 분석

이 글은 주로 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으로 문의하세요.
PHP의 지속적인 사용 : 지구력의 이유PHP의 지속적인 사용 : 지구력의 이유Apr 19, 2025 am 12:23 AM

여전히 인기있는 것은 사용 편의성, 유연성 및 강력한 생태계입니다. 1) 사용 편의성과 간단한 구문은 초보자에게 첫 번째 선택입니다. 2) 웹 개발, HTTP 요청 및 데이터베이스와의 우수한 상호 작용과 밀접하게 통합되었습니다. 3) 거대한 생태계는 풍부한 도구와 라이브러리를 제공합니다. 4) 활성 커뮤니티와 오픈 소스 자연은 새로운 요구와 기술 동향에 맞게 조정됩니다.

PHP 및 Python : 유사점과 차이점을 탐구합니다PHP 및 Python : 유사점과 차이점을 탐구합니다Apr 19, 2025 am 12:21 AM

PHP와 Python은 웹 개발, 데이터 처리 및 자동화 작업에 널리 사용되는 고급 프로그래밍 언어입니다. 1.PHP는 종종 동적 웹 사이트 및 컨텐츠 관리 시스템을 구축하는 데 사용되며 Python은 종종 웹 프레임 워크 및 데이터 과학을 구축하는 데 사용됩니다. 2.PHP는 Echo를 사용하여 콘텐츠를 출력하고 Python은 인쇄를 사용합니다. 3. 객체 지향 프로그래밍을 지원하지만 구문과 키워드는 다릅니다. 4. PHP는 약한 유형 변환을 지원하는 반면, 파이썬은 더 엄격합니다. 5. PHP 성능 최적화에는 Opcache 및 비동기 프로그래밍 사용이 포함되며 Python은 Cprofile 및 비동기 프로그래밍을 사용합니다.

PHP와 Python : 다른 패러다임이 설명되었습니다PHP와 Python : 다른 패러다임이 설명되었습니다Apr 18, 2025 am 12:26 AM

PHP는 주로 절차 적 프로그래밍이지만 객체 지향 프로그래밍 (OOP)도 지원합니다. Python은 OOP, 기능 및 절차 프로그래밍을 포함한 다양한 패러다임을 지원합니다. PHP는 웹 개발에 적합하며 Python은 데이터 분석 및 기계 학습과 같은 다양한 응용 프로그램에 적합합니다.

PHP와 Python : 그들의 역사에 깊은 다이빙PHP와 Python : 그들의 역사에 깊은 다이빙Apr 18, 2025 am 12:25 AM

PHP는 1994 년에 시작되었으며 Rasmuslerdorf에 의해 개발되었습니다. 원래 웹 사이트 방문자를 추적하는 데 사용되었으며 점차 서버 측 스크립팅 언어로 진화했으며 웹 개발에 널리 사용되었습니다. Python은 1980 년대 후반 Guidovan Rossum에 의해 개발되었으며 1991 년에 처음 출시되었습니다. 코드 가독성과 단순성을 강조하며 과학 컴퓨팅, 데이터 분석 및 기타 분야에 적합합니다.

PHP와 Python 중에서 선택 : 가이드PHP와 Python 중에서 선택 : 가이드Apr 18, 2025 am 12:24 AM

PHP는 웹 개발 및 빠른 프로토 타이핑에 적합하며 Python은 데이터 과학 및 기계 학습에 적합합니다. 1.PHP는 간단한 구문과 함께 동적 웹 개발에 사용되며 빠른 개발에 적합합니다. 2. Python은 간결한 구문을 가지고 있으며 여러 분야에 적합하며 강력한 라이브러리 생태계가 있습니다.

PHP 및 프레임 워크 : 언어 현대화PHP 및 프레임 워크 : 언어 현대화Apr 18, 2025 am 12:14 AM

PHP는 현대화 프로세스에서 많은 웹 사이트 및 응용 프로그램을 지원하고 프레임 워크를 통해 개발 요구에 적응하기 때문에 여전히 중요합니다. 1.PHP7은 성능을 향상시키고 새로운 기능을 소개합니다. 2. Laravel, Symfony 및 Codeigniter와 같은 현대 프레임 워크는 개발을 단순화하고 코드 품질을 향상시킵니다. 3. 성능 최적화 및 모범 사례는 응용 프로그램 효율성을 더욱 향상시킵니다.

PHP의 영향 : 웹 개발 및 그 이상PHP의 영향 : 웹 개발 및 그 이상Apr 18, 2025 am 12:10 AM

phphassignificallyimpactedwebdevelopmentandextendsbeyondit

스칼라 유형, 반환 유형, 노조 유형 및 무효 유형을 포함한 PHP 유형의 힌트 작업은 어떻게 작동합니까?스칼라 유형, 반환 유형, 노조 유형 및 무효 유형을 포함한 PHP 유형의 힌트 작업은 어떻게 작동합니까?Apr 17, 2025 am 12:25 AM

PHP 유형은 코드 품질과 가독성을 향상시키기위한 프롬프트입니다. 1) 스칼라 유형 팁 : PHP7.0이므로 int, float 등과 같은 기능 매개 변수에 기본 데이터 유형을 지정할 수 있습니다. 2) 반환 유형 프롬프트 : 기능 반환 값 유형의 일관성을 확인하십시오. 3) Union 유형 프롬프트 : PHP8.0이므로 기능 매개 변수 또는 반환 값에 여러 유형을 지정할 수 있습니다. 4) Nullable 유형 프롬프트 : NULL 값을 포함하고 널 값을 반환 할 수있는 기능을 포함 할 수 있습니다.

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

뜨거운 도구

PhpStorm 맥 버전

PhpStorm 맥 버전

최신(2018.2.1) 전문 PHP 통합 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Eclipse용 SAP NetWeaver 서버 어댑터

Eclipse용 SAP NetWeaver 서버 어댑터

Eclipse를 SAP NetWeaver 애플리케이션 서버와 통합합니다.

에디트플러스 중국어 크랙 버전

에디트플러스 중국어 크랙 버전

작은 크기, 구문 강조, 코드 프롬프트 기능을 지원하지 않음

DVWA

DVWA

DVWA(Damn Vulnerable Web App)는 매우 취약한 PHP/MySQL 웹 애플리케이션입니다. 주요 목표는 보안 전문가가 법적 환경에서 자신의 기술과 도구를 테스트하고, 웹 개발자가 웹 응용 프로그램 보안 프로세스를 더 잘 이해할 수 있도록 돕고, 교사/학생이 교실 환경 웹 응용 프로그램에서 가르치고 배울 수 있도록 돕는 것입니다. 보안. DVWA의 목표는 다양한 난이도의 간단하고 간단한 인터페이스를 통해 가장 일반적인 웹 취약점 중 일부를 연습하는 것입니다. 이 소프트웨어는