>백엔드 개발 >PHP 튜토리얼 >Laravel Under The Hood - 약간의 매크로

Laravel Under The Hood - 약간의 매크로

DDD
DDD원래의
2024-11-07 13:13:03401검색

Laravel Under The Hood - A Little Bit of Macros

안녕하세요 ?

컬렉션이나 문자열 도우미에 존재하지 않는 메서드를 얼마나 자주 원하셨나요? 방법을 연결하기 시작했지만 그 중 하나가 누락된 것으로 판명되면 벽에 부딪히게 됩니다. 솔직히 이해할 수 있습니다. 아시다시피 프레임워크는 모든 것에 적용됩니다. 나는 이런 상황에 여러 번 직면했습니다. 매번 프레임워크를 확장하는 방법에 대해 알아보기 전에 내가 확장하려는 것이 매크로 가능한 것인지 확인합니다. 그런데 그게 무슨 뜻인가요? 이것이 바로 우리가 탐구할 내용입니다!

뭐야 매크로야? ?

다음 JWT가 있다고 가정해 보겠습니다.

$jwt = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c';

그리고 헤더를 추출해야 합니다.

str($jwt)
    ->before('.')
    ->fromBase64()
    ->fromJson(); // does not exist ?

//  BadMethodCallException  Method Illuminate\Support\Stringable::fromJson does not exist.

fromJson()이 존재하지 않나요? 물론 간단하게 다음과 같이 할 수도 있습니다.

json_decode(str($jwt)->before('.')->fromBase64());

그런데 그게 무슨 재미가 있나요? 게다가 내 글이라니?

따라서 Stringable 클래스를 확장할 방법이 필요합니다. 이를 수행하는 방법은 몇 가지가 있지만 Laravel은 개발자가 사용자 정의 메소드를 추가하고 싶어할 수도 있다는 것을 미리 알고 있었기 때문에 클래스를 매크로 가능 또는 확장 가능하게 만들었습니다.

IlluminateSupportStringable 클래스를 살펴보면 Macroable 특성을 사용하는 것을 볼 수 있습니다.

계속해서 수업을 연장해 보겠습니다. AppServiceProvider에서 다음을 추가합니다.

<?php

namespace App\Providers;

use Illuminate\Support\Stringable;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
{
    public function boot(): void
    {
        Stringable::macro('fromJson', function (bool $associative = true) {
            return json_decode($this->value, $associative);
        });
    }
}

이제 코드를 다시 실행해 보겠습니다.

str($jwt)
    ->before('.')
    ->fromBase64()
    ->fromJson();

// ["alg" => "HS256", "typ" => "JWT"]

완벽하게 작동하나요? 하지만 이제 여러분은 이것이 어떻게 작동했는지 궁금할 것입니다. $this->value는 정확히 무엇인가요? 해리포터에서는 무슨 일이 벌어지고 있는 걸까요?

마법을 공개하시나요?

우리는 Stringable 클래스가 매크로() 메소드를 제공하는 Macroable 특성을 사용한다는 것을 알고 있습니다. 그 기능을 자세히 살펴보겠습니다.

// src/Illuminate/Macroable/Traits/Macroable.php

/**
 * Register a custom macro.
 *
 * @param  string  $name
 * @param  object|callable  $macro
 *
 * @param-closure-this static  $macro
 *
 * @return void
 */
public static function macro($name, $macro)
{
    static::$macros[$name] = $macro;
}

매우 간단합니다. 콜백을 정적 매크로 배열에 저장하기만 하면 됩니다. 이제 특성을 더 자세히 조사하면 존재하지 않는 메서드가 호출될 때마다 트리거되는 __call 메서드를 찾을 수 있습니다. 우리의 경우에는 fromJson()입니다. 자세히 살펴보겠습니다:

/**
 * Dynamically handle calls to the class.
 *
 * @param  string  $method
 * @param  array  $parameters
 * @return mixed
 *
 * @throws \BadMethodCallException
 */
public function __call($method, $parameters)
{
    if (! static::hasMacro($method)) {
        throw new BadMethodCallException(sprintf(
            'Method %s::%s does not exist.', static::class, $method
        ));
    }

    $macro = static::$macros[$method];

    if ($macro instanceof Closure) {
        $macro = $macro->bindTo($this, static::class);
    }

    return $macro(...$parameters);
}

먼저 fromJson()의 경우와 같이 매크로가 등록되었는지 확인한 다음 매크로 배열에서 콜백(또는 개체)을 가져옵니다. 이제 마술 트릭을 위해, 매크로가 클로저(우리의 경우처럼)라면, 이는 기본적으로 $this가 첫 번째 인수로 전달된 모든 것을 참조해야 한다는 것을 클로저에게 알려주는 바인딩To()를 호출합니다. 이 경우 $value 속성을 갖는 Stringable 인스턴스입니다.

// $this here is the stringable
// $this inside the closure is now referencing the stringable class
$macro->bindTo($this, static::class);

그리고 이것이 바로 우리가 $this->value를 수행할 수 있는 이유입니다.

우리는 더 잘할 수 있습니다: Mixins ?

한 가지 더 보여드리고 싶은 게 있어요! 동일한 클래스를 몇 번 확장하면 서비스 제공자가 매우 빨리 지저분해질 수 있습니다. 모든 사용자 정의 매크로를 Mixin이라는 클래스로 추출할 수 있습니다.

StringableMixin을 만들어 보겠습니다.

$jwt = 'eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiaWF0IjoxNTE2MjM5MDIyfQ.SflKxwRJSMeKKF2QT4fwpMeJf36POk6yJV_adQssw5c';

이제 AppServiceProvider에서 다음 믹스인을 등록할 수 있습니다.

str($jwt)
    ->before('.')
    ->fromBase64()
    ->fromJson(); // does not exist ?

//  BadMethodCallException  Method Illuminate\Support\Stringable::fromJson does not exist.

그리고 그게 다입니다! 이제 다음을 수행할 수 있습니다.

json_decode(str($jwt)->before('.')->fromBase64());

기본적으로 동일하지만 조금 더 깔끔합니다.

이것이 어떻게 작동하는지 궁금하다면 Macroable 특성의 mixin() 메서드가 리플렉션 API를 사용합니다. Mixin 클래스에서 모든 공개 메소드를 검색하고 각각이 클로저를 반환할 것으로 예상한 다음 이전에 본 것처럼 클로저를 매크로로 등록합니다.

불쌍한 IDE?

글쎄요, 보시다시피 많은 마법이 일어나고 있으며 IDE는 정의된 매크로에 대해 알지 못합니다. 팀 내에서 작업하는 경우 다른 개발자도 이러한 매크로에 대해 알지 못하므로 좋지 않습니다. 다행히도 이를 도와주는 도구가 있습니다. 무료 오픈 소스 옵션은 Laravel IDE 도우미 패키지입니다.

패키지를 설치하고 _ide_helper.php 파일을 생성하면 됩니다.

그리고 그렇게 끝난다..

우리의 예는 매우 간단하지만 Laravel에서 제공하는 대부분의 일반적인 클래스는 매크로 가능하므로 이보다 훨씬 더 많은 매크로를 푸시할 수 있습니다. 예를 들어, 새로운 apiResponse() 매크로를 추가하거나 앱의 로직에서 매우 일반적이고 필요 이상으로 반복된다고 생각되는 모든 것을 추가할 수 있습니다. 하지만 과용하지 마세요. 매크로는 새로운 복잡성을 추가하며 팀으로 작업할 때 혼란스러울 수 있습니다.

그래서, 프레임워크 자체에는 없지만 애플리케이션에는 뭔가 빠졌다고 느낄 때마다 매크로를 사용하세요.

위 내용은 Laravel Under The Hood - 약간의 매크로의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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