development:software-architecture:design-patterns:proxy
This is an old revision of the document!
Proxy
Tại sao muốn kiểm soát quyền truy cập vào một đối tượng?
Đây là một ví dụ: bạn có một đối tượng database lớn tiêu thụ một lượng lớn tài nguyên hệ thống. Bạn cần nó theo thời gian, nhưng không phải lúc nào cũng cần dùng đến.
Mẫu Proxy gợi ý rằng bạn tạo một lớp proxy mới có cùng giao diện với đối tượng dịch vụ gốc. Sau đó, bạn cập nhật ứng dụng của mình để nó chuyển đối tượng proxy cho tất cả các máy khách của đối tượng gốc. Khi nhận được yêu cầu từ máy khách, proxy tạo một đối tượng dịch vụ thực và phân công tất cả công việc cho đối tượng đó.
Trong phần mềm hệ thống proxy chủ yếu dùng để kiểm soát, bảo vệ & điều hướng dữ liệu đầu vào.
Ví dụ 1:
- LeaderObject.php
<?php namespace App\Utils\Object; /** * LeaderObject */ class LeaderObject { /** * reviceRequestOffer * * @param mixed $offer * @return string */ public function reviceRequestOffer($offer) { return "Offer {$offer} is ok!"; } }
- SecretaryProxy.php
<?php namespace App\Utils\Proxy; use App\Utils\Object\LeaderObject; /** * SecretaryProxy */ class SecretaryProxy { /** * leader * * @var mixed */ private $leader; /** * __construct * * @param mixed $leader * @return void */ public function __construct(LeaderObject $leader) { $this->leader = $leader; } /** * reviceRequestOffer * * @param mixed $offer * @return string */ public function reviceRequestOffer($offer) { // ... return $this->leader->reviceRequestOffer($offer); } }
- DeveloperObject.php
<?php namespace App\Utils\Object; use App\Utils\Proxy\SecretaryProxy; /** * DeveloperObject */ class DeveloperObject { /** * applyFor * * @param mixed $secretary * @param mixed $offer * @return string */ public function applyFor(SecretaryProxy $secretary, $offer) { return $secretary->reviceRequestOffer($offer); } }
- ProxyCommand.php
<?php namespace App\Console\Commands\Test; use App\Utils\Object\DeveloperObject; use App\Utils\Object\LeaderObject; use App\Utils\Proxy\SecretaryProxy; use Illuminate\Console\Command; /** * Class ProxyCommand * @package Modules\Web\Console\Commands */ class ProxyCommand extends Command { /** * The console command name. * * @var string */ protected $signature = 'test:proxy'; public function handle(){ $leader = new LeaderObject(); $developer = new DeveloperObject(); $result = $developer->applyFor(new SecretaryProxy($leader), '50M'); $this->info($result); } }
Ví dụ 2:
<?php namespace RefactoringGuru\Proxy\RealWorld; /** * The Subject interface describes the interface of a real object. * * The truth is that many real apps may not have this interface clearly defined. * If you're in that boat, your best bet would be to extend the Proxy from one * of your existing application classes. If that's awkward, then extracting a * proper interface should be your first step. */ interface Downloader { public function download(string $url): string; } /** * The Real Subject does the real job, albeit not in the most efficient way. * When a client tries to download the same file for the second time, our * downloader does just that, instead of fetching the result from cache. */ class SimpleDownloader implements Downloader { public function download(string $url): string { echo "Downloading a file from the Internet.\n"; $result = file_get_contents($url); echo "Downloaded bytes: " . strlen($result) . "\n"; return $result; } } /** * The Proxy class is our attempt to make the download more efficient. It wraps * the real downloader object and delegates it the first download calls. The * result is then cached, making subsequent calls return an existing file * instead of downloading it again. * * Note that the Proxy MUST implement the same interface as the Real Subject. */ class CachingDownloader implements Downloader { /** * @var SimpleDownloader */ private $downloader; /** * @var string[] */ private $cache = []; public function __construct(SimpleDownloader $downloader) { $this->downloader = $downloader; } public function download(string $url): string { if (!isset($this->cache[$url])) { echo "CacheProxy MISS. "; $result = $this->downloader->download($url); $this->cache[$url] = $result; } else { echo "CacheProxy HIT. Retrieving result from cache.\n"; } return $this->cache[$url]; } } /** * The client code may issue several similar download requests. In this case, * the caching proxy saves time and traffic by serving results from cache. * * The client is unaware that it works with a proxy because it works with * downloaders via the abstract interface. */ function clientCode(Downloader $subject) { // ... $result = $subject->download("http://example.com/"); // Duplicate download requests could be cached for a speed gain. $result = $subject->download("http://example.com/"); // ... } echo "Executing client code with real subject:\n"; $realSubject = new SimpleDownloader(); clientCode($realSubject); echo "\n"; echo "Executing the same client code with a proxy:\n"; $proxy = new CachingDownloader($realSubject); clientCode($proxy);
Tham khảo
development/software-architecture/design-patterns/proxy.1723708586.txt.gz · Last modified: 2024/08/15 07:56 by tungnt