development:software-architecture:design-patterns:proxy
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
development:software-architecture:design-patterns:proxy [2024/08/15 06:56] – tungnt | development:software-architecture:design-patterns:proxy [2024/08/19 10:52] (current) – [Proxy] tungnt | ||
---|---|---|---|
Line 10: | Line 10: | ||
{{ : | {{ : | ||
+ | |||
+ | 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:** | ||
+ | |||
+ | <file php LeaderObject.php> | ||
+ | <?php | ||
+ | |||
+ | namespace App\Utils\Object; | ||
+ | |||
+ | /** | ||
+ | * LeaderObject | ||
+ | */ | ||
+ | class LeaderObject | ||
+ | { | ||
+ | /** | ||
+ | * reviceRequestOffer | ||
+ | * | ||
+ | * @param | ||
+ | * @return string | ||
+ | */ | ||
+ | public function reviceRequestOffer($offer) | ||
+ | { | ||
+ | return "Offer {$offer} is ok!"; | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | <file php SecretaryProxy.php> | ||
+ | <?php | ||
+ | |||
+ | namespace App\Utils\Proxy; | ||
+ | use App\Utils\Object\LeaderObject; | ||
+ | |||
+ | /** | ||
+ | * SecretaryProxy | ||
+ | */ | ||
+ | class SecretaryProxy | ||
+ | { | ||
+ | /** | ||
+ | * leader | ||
+ | * | ||
+ | * @var mixed | ||
+ | */ | ||
+ | private $leader; | ||
+ | /** | ||
+ | * __construct | ||
+ | * | ||
+ | * @param | ||
+ | * @return void | ||
+ | */ | ||
+ | public function __construct(LeaderObject $leader) { | ||
+ | $this-> | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * reviceRequestOffer | ||
+ | * | ||
+ | * @param | ||
+ | * @return string | ||
+ | */ | ||
+ | public function reviceRequestOffer($offer) | ||
+ | { | ||
+ | // ... | ||
+ | |||
+ | return $this-> | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | <file php DeveloperObject.php> | ||
+ | <?php | ||
+ | |||
+ | namespace App\Utils\Object; | ||
+ | use App\Utils\Proxy\SecretaryProxy; | ||
+ | |||
+ | /** | ||
+ | * DeveloperObject | ||
+ | */ | ||
+ | class DeveloperObject | ||
+ | { | ||
+ | /** | ||
+ | * applyFor | ||
+ | * | ||
+ | * @param | ||
+ | * @param | ||
+ | * @return string | ||
+ | */ | ||
+ | | ||
+ | { | ||
+ | | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | <file php 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 = ' | ||
+ | |||
+ | public function handle(){ | ||
+ | $leader = new LeaderObject(); | ||
+ | $developer = new DeveloperObject(); | ||
+ | |||
+ | $result = $developer-> | ||
+ | |||
+ | $this-> | ||
+ | } | ||
+ | |||
+ | } | ||
+ | </ | ||
+ | |||
+ | **Ví dụ 2:** | ||
+ | |||
+ | <file php> | ||
+ | <?php | ||
+ | |||
+ | namespace RefactoringGuru\Proxy\Conceptual; | ||
+ | |||
+ | /** | ||
+ | * The Subject interface declares common operations for both RealSubject and the | ||
+ | * Proxy. As long as the client works with RealSubject using this interface, | ||
+ | * you'll be able to pass it a proxy instead of a real subject. | ||
+ | */ | ||
+ | interface Subject | ||
+ | { | ||
+ | public function request(): void; | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * The RealSubject contains some core business logic. Usually, RealSubjects are | ||
+ | * capable of doing some useful work which may also be very slow or sensitive - | ||
+ | * e.g. correcting input data. A Proxy can solve these issues without any | ||
+ | * changes to the RealSubject' | ||
+ | */ | ||
+ | class RealSubject implements Subject | ||
+ | { | ||
+ | public function request(): void | ||
+ | { | ||
+ | echo " | ||
+ | } | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * The Proxy has an interface identical to the RealSubject. | ||
+ | */ | ||
+ | class Proxy implements Subject | ||
+ | { | ||
+ | /** | ||
+ | * @var RealSubject | ||
+ | */ | ||
+ | private $realSubject; | ||
+ | |||
+ | /** | ||
+ | * The Proxy maintains a reference to an object of the RealSubject class. It | ||
+ | * can be either lazy-loaded or passed to the Proxy by the client. | ||
+ | */ | ||
+ | public function __construct(RealSubject $realSubject) | ||
+ | { | ||
+ | $this-> | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * The most common applications of the Proxy pattern are lazy loading, | ||
+ | * caching, controlling the access, logging, etc. A Proxy can perform one of | ||
+ | * these things and then, depending on the result, pass the execution to the | ||
+ | * same method in a linked RealSubject object. | ||
+ | */ | ||
+ | public function request(): void | ||
+ | { | ||
+ | if ($this-> | ||
+ | $this-> | ||
+ | $this-> | ||
+ | } | ||
+ | } | ||
+ | |||
+ | private function checkAccess(): | ||
+ | { | ||
+ | // Some real checks should go here. | ||
+ | echo " | ||
+ | |||
+ | return true; | ||
+ | } | ||
+ | |||
+ | private function logAccess(): | ||
+ | { | ||
+ | echo " | ||
+ | } | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * The client code is supposed to work with all objects (both subjects and | ||
+ | * proxies) via the Subject interface in order to support both real subjects and | ||
+ | * proxies. In real life, however, clients mostly work with their real subjects | ||
+ | * directly. In this case, to implement the pattern more easily, you can extend | ||
+ | * your proxy from the real subject' | ||
+ | */ | ||
+ | function clientCode(Subject $subject) | ||
+ | { | ||
+ | // ... | ||
+ | |||
+ | $subject-> | ||
+ | |||
+ | // ... | ||
+ | } | ||
+ | |||
+ | echo " | ||
+ | $realSubject = new RealSubject(); | ||
+ | clientCode($realSubject); | ||
+ | |||
+ | echo " | ||
+ | |||
+ | echo " | ||
+ | $proxy = new Proxy($realSubject); | ||
+ | clientCode($proxy); | ||
+ | </ | ||
+ | |||
+ | **Ví dụ 3:** | ||
+ | |||
+ | <file php> | ||
+ | <?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 " | ||
+ | $result = file_get_contents($url); | ||
+ | echo " | ||
+ | |||
+ | 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-> | ||
+ | } | ||
+ | |||
+ | public function download(string $url): string | ||
+ | { | ||
+ | if (!isset($this-> | ||
+ | echo " | ||
+ | $result = $this-> | ||
+ | $this-> | ||
+ | } else { | ||
+ | echo " | ||
+ | } | ||
+ | return $this-> | ||
+ | } | ||
+ | } | ||
+ | |||
+ | /** | ||
+ | * 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-> | ||
+ | |||
+ | // Duplicate download requests could be cached for a speed gain. | ||
+ | |||
+ | $result = $subject-> | ||
+ | |||
+ | // ... | ||
+ | } | ||
+ | |||
+ | echo " | ||
+ | $realSubject = new SimpleDownloader(); | ||
+ | clientCode($realSubject); | ||
+ | |||
+ | echo " | ||
+ | |||
+ | echo " | ||
+ | $proxy = new CachingDownloader($realSubject); | ||
+ | clientCode($proxy); | ||
+ | </ | ||
+ | |||
+ | ===== Tham khảo ===== | ||
+ | |||
+ | * https:// | ||
+ | * https:// | ||
+ | |||
+ |
development/software-architecture/design-patterns/proxy.1723705009.txt.gz · Last modified: 2024/08/15 06:56 by tungnt