Symfony - Kontainer Layanan
Dalam aplikasi apa pun, objek cenderung meningkat seiring dengan pertumbuhan aplikasi. Saat objek meningkat, ketergantungan antar objek juga meningkat. Ketergantungan objek perlu ditangani dengan benar untuk aplikasi yang berhasil.
Seperti yang didiskusikan di bab Komponen, Symfony menyediakan komponen yang mudah dan efisien, DependencyInjectionuntuk menangani ketergantungan objek. Sebuah wadah layanan adalah wadah objek dengan ketergantungan yang diselesaikan dengan benar di antara mereka. Mari kita pelajari cara menggunakan komponen DependencyInjection di bab ini.
Mari kita buat file Greeterkelas. Tujuan kelas Greeter adalah untuk menyapa pengguna seperti yang ditunjukkan pada contoh berikut.
$greeter = new Greeter('Hi');
$greeter->greet('Jon'); // print "Hi, Jon"
Kode lengkap dari kelas Greeter adalah sebagai berikut.
class Greeter {
private $greetingText;
public function __construct($greetingText) {
$this->greetingText = $greetingText;
}
public function greet($name) {
echo $this->greetingText . ", " . $name . "\r\n";
}
}
Sekarang, mari kita tambahkan kelas Greeter ke service container. Symfony menyediakanContainerBuilderuntuk membuat wadah baru. Setelah penampung dibuat, kelas Penyapa dapat didaftarkan ke dalamnya menggunakan metode register penampung.
use Symfony\Component\DependencyInjection\ContainerBuilder;
$container = new ContainerBuilder();
$container
->register('greeter', 'Greeter')
->addArgument('Hi');
Di sini, kami telah menggunakan argumen statis untuk menentukan teks salam, Hai. Symfony juga menyediakan pengaturan parameter yang dinamis. Untuk menggunakan parameter dinamis, kita perlu memilih nama dan menetapkannya antara% dan parameter dapat disetel menggunakan penampungsetParameter metode.
$container = new ContainerBuilder();
$container
->register('greeter', 'Greeter')
->addArgument('%greeter.text%');
$container->setParameter('greeter.text', 'Hi');
Kami telah mendaftarkan kelas Greeter dengan pengaturan yang tepat. Sekarang, kita dapat meminta container untuk menyediakan objek Greeter yang dikonfigurasi dengan benar menggunakan containerget metode.
$greeter = $container->get('greeter');
$greeter->greet('Jon'); // prints "Hi, Jon"
Kami telah berhasil mendaftarkan kelas, Greeter ke dalam wadah, mengambilnya dari wadah dan menggunakannya. Sekarang, mari kita buat kelas lainUser, yang menggunakan kelas Greeter dan melihat cara mendaftarkannya.
class User {
private $greeter;
public $name;
public $age;
public function setGreeter(\Greeter $greeter) {
$this->greeter = $greeter;
}
public function greet() {
$this->greeter->greet($this->name);
}
}
Kelas User mendapatkan kelas Greeter menggunakan salah satu metode penyetelnya,setGreeter. Untuk skenario ini, Symfony menyediakan metode,addMethodCall dan sebuah kelas, Reference untuk merujuk kelas lain seperti yang ditunjukkan pada kode berikut.
use Symfony\Component\DependencyInjection\Reference;
$container
->register('user', 'User')
->addMethodCall('setGreeter', array(new Reference('greeter')));
Akhirnya, kami telah mendaftarkan dua kelas, Greeter dan Usermemiliki hubungan yang kuat di antara mereka. Sekarang, kita dapat dengan aman mengambil objek User dengan class Greeter yang dikonfigurasi dengan benar dari container seperti yang ditunjukkan pada kode berikut.
$container->setParameter('greeter.text', 'Hi');
$user = $container->get('user');
$user->name = "Jon";
$user->age = 20;
$user->greet(); // Prints "Hi, Jon"
Kami telah melihat cara mengonfigurasi objek dalam wadah menggunakan PHP itu sendiri. Symfony juga menyediakan mekanisme lain. Mereka adalah file konfigurasi XML dan YAML. Mari kita lihat cara mengonfigurasi container menggunakan YAML. Untuk ini, instalsymfony/config dan symfony/yaml komponen bersama dengan symfony/dependency-injection komponen.
cd /path/to/dir
mkdir dependency-injection-example
cd dependency-injection-example
composer require symfony/dependency-injection
composer require symfony/config
composer require symfony/yaml
Konfigurasi YAML akan ditulis dalam file terpisah, services.yml. Konfigurasi YAML terdiri dari dua bagian,parameters dan services. Bagian parameter mendefinisikan semua parameter yang diperlukan. Bagian layanan mendefinisikan semua objek. Bagian layanan selanjutnya dibagi menjadi beberapa bagian yaitu,class, arguments, dan calls. Kelas menentukan kelas yang sebenarnya. Argumen menentukan argumen konstruktor. Terakhir, panggilan menentukan metode penyetel. Kelas lain dapat dirujuk menggunakan simbol @, @greeter.
parameters:
greeter.text: 'Hello'
services:
greeter:
class: Greeter
arguments: ['%greeter.text%']
user:
class: User
calls:
- [setGreeter, ['@greeter']]
Sekarang, services.yml dapat dimuat dan dikonfigurasi menggunakan FileLoader dan YamlFileLoader seperti yang ditunjukkan pada kode berikut.
use Symfony\Component\Config\FileLocator;
use Symfony\Component\DependencyInjection\Loader\YamlFileLoader;
$yamlContainer = new ContainerBuilder();
$loader = new YamlFileLoader($yamlContainer, new FileLocator(__DIR__));
$loader->load('services.yml');
$yamlUser = $yamlContainer->get('user');
$yamlUser->name = "Jon";
$yamlUser->age = 25;
$yamlUser->greet();
Daftar kode lengkapnya adalah sebagai berikut.
main.php
<?php
require __DIR__ . '/vendor/autoload.php';
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\Config\FileLocator;
use Symfony\Component\DependencyInjection\Loader\YamlFileLoader;
use Symfony\Component\DependencyInjection\Reference;
class Greeter {
private $greetingText;
public function __construct($greetingText) {
$this->greetingText = $greetingText;
}
public function greet($name) {
echo $this->greetingText . ", " . $name . "\r\n";
}
}
class User {
private $greeter;
public $name;
public $age;
public function setGreeter(\Greeter $greeter) {
$this->greeter = $greeter;
}
public function greet() {
$this->greeter->greet($this->name);
}
}
$container = new ContainerBuilder();
$container
->register('greeter', 'Greeter')
->addArgument('%greeter.text%');
$container
->register('user', 'User')
->addMethodCall('setGreeter', array(new Reference('greeter')));
$container->setParameter('greeter.text', 'Hi');
$greeter = $container->get('greeter');
$greeter->greet('Jon');
$user = $container->get('user');
$user->name = "Jon";
$user->age = 20;
$user->greet();
$yamlContainer = new ContainerBuilder();
$loader = new YamlFileLoader($yamlContainer, new FileLocator(__DIR__));
$loader->load('services.yml');
$yamlHello = $yamlContainer->get('greeter');
$yamlHello->greet('Jon');
$yamlUser = $yamlContainer->get('user');
$yamlUser->name = "Jon";
$yamlUser->age = 25;
$yamlUser->greet();
?>
services.yml
parameters:
greeter.text: 'Hello'
services:
greeter:
class: Greeter
arguments: ['%greeter.text%']
user:
class: User
calls:
- [setGreeter, ['@greeter']]
Framework web Symfony menggunakan komponen injeksi dependensi secara ekstensif. Semua komponen terikat oleh wadah layanan terpusat. Framework web Symfony memperlihatkan container di semua fileController melalui containerProperti. Kita bisa mendapatkan semua objek terdaftar di dalamnya, katakanlah logger, mailer, dll., Melalui itu.
$logger = $this->container->get('logger');
$logger->info('Hi');
Untuk menemukan objek yang terdaftar di wadah, gunakan perintah berikut.
cd /path/to/app
php bin/console debug:container
Ada sekitar 200+ objek di hello aplikasi web yang dibuat di bab instalasi.