قابلیت ارتقای مشخصه سازنده در PHP — به زبان ساده

۴۸ بازدید
آخرین به‌روزرسانی: ۰۸ مهر ۱۴۰۲
زمان مطالعه: ۴ دقیقه
قابلیت ارتقای مشخصه سازنده در PHP — به زبان ساده

به زودی شاهد انتشار رسمی جدیدترین نسخه از زبان برنامه‌نویسی پی‌اچ‌پی‌ به نام PHP 8 خواهیم بود که قابلیت‌ها و امکانات تازه زیادی را معرفی می‌کند. یکی از این امکانات مهم «ارتقای مشخصه سازنده» (Constructor Property Promotion) نام دارد. این قابلیت موجب می‌شود که بخش زیادی از کدهای تکراری که در زمان ساخت اشیای ساده مانند «اشیای مقداری» (VO) و «اشیای انتقال داده» (DTO) باید می‌نوشتیم، کاهش یابند. در این مقاله به توضیح قابلیت ارتقای مشخصه سازنده در PHP می‌پردازیم.

به طور خلاصه ارتقای مشخصه موجب می‌شود که بتوانیم فیلدهای کلاس، تعریف سازنده و انتساب‌های متغیر را در یک ساختار منفرد در لیست پارامتر سازنده ترکیب کنیم. به این ترتیب به جای این که به روش قدیم کدی مانند زیر بنویسیم:

1class CustomerDTO
2{
3    public string $name;
4
5    public string $email;
6
7    public DateTimeImmutable $birth_date;
8
9    public function __construct(
10        string $name, 
11        string $email, 
12        DateTimeImmutable $birth_date
13    ) {
14        $this->name = $name;
15        $this->email = $email;
16        $this->birth_date = $birth_date;
17    }
18}

می‌توانیم از کدی مانند زیر استفاده کنیم:

1class CustomerDTO
2{
3    public function __construct(
4        public string $name, 
5        public string $email, 
6        public DateTimeImmutable $birth_date,
7    ) {}
8}

طرز کار قابلیت ارتقای مشخصه سازنده در PHP چگونه است؟

ایده اصلی این قابلیت کاملاً ساده است. ما می‌خواهیم همه مشخصه‌های کلاس، انتساب‌های مقادیر و پیشوندهای پارامتر سازنده مانند public ،protected یا private را حذف کنیم.

PHP ساختار جدیدی دریافت می‌کند و آن را در پس‌زمینه پیش از اجرای عملی کد، به ساختار قدیمی تبدیل می‌کند.

بنابراین در ساختار جدید از کدی مانند زیر استفاده کنیم:

1class MyDTO
2{
3    public function __construct(
4        public string $name = 'Brent',
5    ) {}
6}

که جایگزین کد قدیمی زیر شده است:

1class MyDTO
2{
3    public string $name;
4
5    public function __construct(
6        string $name = 'Brent'
7    ) {
8        $this->name = $name;
9    }
10}

توجه کنید که مقدار پیش‌فرض روی مشخصه کلاس تعیین نشده است، بلکه روی آرگومان متد در سازنده تعیین گشته است.

قابلیت ارتقای مشخصه سازنده در PHP

مشخصات مشخصه ارتقا یافته

در این بخش به بررسی مشخصه‌های این قابلیت جدید و کارهایی که می‌توانیم یا نمی‌توانیم با آن انجام دهیم می‌پردازیم. در این خصوص موارد ظریف زیادی وجود دارند که باید به آن‌ها توجه داشته باشیم.

کاربرد صرف در سازنده‌ها

توجه کنید که مشخصه‌های ارتقا یافته را تنها می‌توان در سازنده‌ها (Constructors) استفاده کرد. این موضوع شاید بدیهی به نظر برسد، اما برای روشن‌تر شدن موضوع لازم بود که دوباره یادآوری کنیم.

امکان استفاده از هر دو روش وجود ندارد

شما نمی‌توانید یک مشخصه کلاس و یک مشخصه ارتقا یافته را با نام یکسان اعلان کنید. این موضوع منطقی هم نیست، زیرا مشخصه ارتقا یافته در عمل در زمان اجرا به مشخصه کلاس ترجمه می‌شود:

1class MyClass
2{
3    public string $a;
4
5    public function __construct(
6        public string $a,
7    ) {}
8}

مشخصه‌های بدون نوع مجاز هستند

امکان ارتقای مشخصه‌های فاقد نوع وجود دارد، با این حال با توجه به رویکردهای PHP مدرن، بهتر است که هر چیزی را نوع‌بندی کنید.

1class MyDTO
2{
3    public function __construct(
4        public $untyped,
5    ) {}
6}

مقادیر ساده پیش‌فرض

مشخصه‌های ارتقا یافته می‌توانند مقادیر پیش‌فرض داشته باشند، اما عبارت‌هایی مانند …new مجاز نیستند.

1public function __construct(
2    public string $name = 'Brent',
3    public DateTimeImmutable $date = new DateTimeImmutable(),
4) {}

ترکیب مشخصه‌های ارتقا یافته و معمولی

همه مشخصه‌های سازنده نباید ارتقا یابند و شما می‌توانید آن‌ها را ترکیب و تطبیق دهید.

1class MyClass
2{
3    public string $b;
4
5    public function __construct(
6        public string $a,
7        string $b,
8    ) {
9        $this->b = $b;
10    }
11}

البته در زمان ترکیب ساختارها باید هوشیار باشید، اگر این کار موجب می‌شود که کد روشن نباشد، بهتر است به جای آن از سازنده معمولی استفاده کنید.

دسترسی به مشخصه‌های ارتقا یافته از بدنه سازنده

شما مجاز به خواندن مشخصه‌های ارتقا یافته از بدنه سازنده هستید. این موضوع می‌تواند در صورت نیاز به بررسی‌های اعتبارسنجی بیشتر، مفید باشد. شما می‌توانید از هر دو نوع متغیرهای لوکال و متغیر وهله‌ای استفاده کنید و هر دو نوع به درستی کار می‌کنند.

1public function __construct(
2    public int $a,
3    public int $b,
4) {
5    assert($this->a >= 100);
6
7    if ($b >= 0) {
8        throw new InvalidArgumentException('…');
9    }
10}

کامنت‌های مستندسازی روی مشخصه‌های ارتقا یافته

امکان افزودن توضیح‌های مستندسازی به مشخصه‌های ارتقا یافته وجود دارد و همچنان در زمان رفلکشن در دسترس شما قرار دارند.

1class MyClass 
2{
3    public function __construct(
4        /** @var string */
5        public $a,
6    ) {}
7}
1$property = new ReflectionProperty(MyClass::class, 'a');
2$property->getDocComment(); // "/** @var string */"

خصوصیت‌ها

همانند کامنت‌های مستندسازی که در بخش قبل اشاره کردیم، امکان استفاده از «خصوصیت‌ها» (Attributes) روی مشخصه‌های ارتقا یافته وجود دارد. زمانی که این موارد transpile می‌شوند، روی هر دو نوع پارامتر سازنده و همچنین مشخصه کلاس در دسترس قرار می‌گیرند.

1class MyClass
2{
3    public function __construct(
4        #[MyAttribute]
5        public $a,  
6    ) {}
7}

که به حالت زیر transpile می‌شود:

1class MyClass 
2{
3    #[MyAttribute]
4    public $a;
5 
6    public function __construct(
7        #[MyAttribute]
8        $a,
9    ) {
10        $this->a = $a;
11    }
12}

در سازنده‌های مجرد مجاز نیستند

شاید حتی ندانید که نوعی از سازنده‌ها به نام «سازنده‌های مجرد» (Abstract Constructors) ‌وجود دارند، اما به هر حال باید بدانید که مشخصه‌های ارتقا یافته روی این نوع از سازنده‌ها مجاز نیستند.

1abstract class A
2{
3    abstract public function __construct(
4        public string $a,
5    ) {}
6}

روی خصیصه‌ها مجاز هستند

اما از سوی دیگر امکان استفاده از مشخصه‌های ارتقا یافته روی «خصیصه‌ها» (Traits) وجود دارد. این وضعیت منطقی است چون ساختار تبدیل یافته (Transpiled) نیز در خصیصه‌ها مجاز است.

1trait MyTrait
2{
3    public function __construct(
4        public string $a,
5    ) {}
6}

Var پشتیبانی نمی‌شود

توسعه‌دهندگان پیشکسوت و باتجربه PHP ممکن است در گذشته دور از کلیدواژه var برای اعلان متغیرهای کلاس استفاده کرده باشند. باید اشاره کنیم که امکان استفاده از آن در مشخصه‌های ارتقا یافته وجود ندارد. تنها کلیدواژه‌های مجاز شامل موارد public ،protected و private هستند.

1public function __construct(
2    var string $a,
3) {}

پارامترهای نامعین (Variadic) نمی‌توانند ارتقا یابند

از آنجا که نمی‌توانید داده‌ای را به نوعی از «آرایه نوع‌ها» (array of type) تبدیل کنید، امکان ارتقای پارامترهای نامعین نیز وجود ندارد.

1public function __construct(
2    public string ...$a,
3) {}

در این مورد باید از ژنریک‌ها استفاده کنیم.

Reflection برای isPromoted

هر دو گزینه ReflectionProperty و ReflectionParameter اینک دارای یک متد به نام isPromoted هستند که بررسی می‌کند آیا مشخصه کلاس یا پارامتر متد ارتقا یافته است.

وراثت

از آنجا که سازنده‌های PHP لزومی به تبعیت از اعلان سازنده والد خود ندارند، بنابراین به طور مختصر می‌توان اشاره کرد که وراثت در مشخصه‌های ارتقا یافته مجاز است. با این حال اگر نیاز به ارسال مشخصه‌ها از سازنده فرزند به سازنده والد باشد، باید آن‌ها را به صورت دستی ارسال کنید:

1class A
2{
3    public function __construct(
4        public $a,
5    ) {}
6}
7
8class B extends A
9{
10    public function __construct(
11        $a,
12        public $b,    
13    ) {
14        parent::__construct($a);
15    }
16}

سخن پایانی

به این ترتیب به انتهای این مقاله با موضوع مشخصه‌های ارتقا یافته در PHP می‌رسیم. استفاده از این قابلیت جدید PHP 8 موجب سهولت زیادی در فرایند کارها می‌شود و از این رو پیشنهاد می‌کنیم، آن را جایگزین روش قدیمی اعلان مشخصه‌های کلاس بکنید.

بر اساس رای ۰ نفر
آیا این مطلب برای شما مفید بود؟
اگر بازخوردی درباره این مطلب دارید یا پرسشی دارید که بدون پاسخ مانده است، آن را از طریق بخش نظرات مطرح کنید.
منابع:
stitcher
نظر شما چیست؟

نشانی ایمیل شما منتشر نخواهد شد. بخش‌های موردنیاز علامت‌گذاری شده‌اند *