Vzor servisní vrstvy v XOOPS
2.5.x ✅ 4.0.x ✅
:::poznámka[Nejste si jisti, zda je to správný vzor?] Viz Volba vzoru přístupu k datům, kde najdete strom rozhodování porovnávající obslužné programy, úložiště, služby a CQRS. :::
Vzor Service Layer Pattern zapouzdřuje obchodní logiku do vyhrazených tříd služeb a poskytuje jasné oddělení mezi řadiči a vrstvami pro přístup k datům. Tento vzor podporuje opětovnou použitelnost kódu, testovatelnost a udržovatelnost.
Koncept servisní vrstvy
Sekce “Koncept servisní vrstvy”Účel
Sekce “Účel”Servisní vrstva:
- Obsahuje doménovou obchodní logiku
- Koordinuje více úložišť
- Zvládá složité operace
- Řídí transakce
- Provádí validaci a autorizaci
- Poskytuje ovládání na vysoké úrovni
Výhody
Sekce “Výhody”- Opakovaně použitelná obchodní logika napříč více ovladači
- Snadné testování v izolaci
- Centralizovaná implementace obchodních pravidel
- Jasné oddělení zájmů
- Zjednodušený kód ovladače
Dependency Injection
Sekce “Dependency Injection”<?php// Service with injected dependenciesclass UserService{ private $userRepository; private $emailService;
public function __construct( UserRepositoryInterface $userRepository, EmailServiceInterface $emailService ) { $this->userRepository = $userRepository; $this->emailService = $emailService; }
public function registerUser($username, $email, $password) { // Validate $this->validate($username, $email, $password);
// Create user $user = new User(); $user->setUsername($username); $user->setEmail($email); $user->setPassword($password);
// Save $userId = $this->userRepository->save($user);
// Send welcome email $this->emailService->sendWelcome($email, $username);
return $userId; }
private function validate($username, $email, $password) { $errors = [];
if (strlen($username) < 3) { $errors['username'] = 'Username too short'; }
if (!filter_var($email, FILTER_VALIDATE_EMAIL)) { $errors['email'] = 'Invalid email'; }
if (strlen($password) < 6) { $errors['password'] = 'Password too short'; }
if (!empty($errors)) { throw new ValidationException('Invalid input', $errors); } }}?>Servisní kontejner
Sekce “Servisní kontejner”<?phpclass ServiceContainer{ private $services = [];
public function __construct($db) { // Register repositories $this->services['userRepository'] = new UserRepository($db);
// Register services $this->services['userService'] = new UserService( $this->services['userRepository'] ); }
public function get($name) { if (!isset($this->services[$name])) { throw new \InvalidArgumentException("Service not found: $name"); } return $this->services[$name]; }}?>Použití v ovladačích
Sekce “Použití v ovladačích”<?phpclass UserController{ private $userService;
public function __construct(ServiceContainer $container) { $this->userService = $container->get('userService'); }
public function registerAction() { if ($_SERVER['REQUEST_METHOD'] !== 'POST') { return []; }
try { $userId = $this->userService->registerUser( $_POST['username'], $_POST['email'], $_POST['password'] );
return [ 'success' => true, 'userId' => $userId, ]; } catch (ValidationException $e) { return [ 'success' => false, 'errors' => $e->getErrors(), ]; } }}?>Nejlepší postupy
Sekce “Nejlepší postupy”- Každá služba zpracovává jednu doménu
- Služby závisí na rozhraních, nikoli na implementacích
- Použijte vložení konstruktoru pro závislosti
- Služby by měly být testovatelné samostatně
- Vyhazujte výjimky specifické pro doménu
- Služby by neměly záviset na podrobnostech požadavku HTTP
- Udržujte služby zaměřené a soudržné
Související dokumentace
Sekce “Související dokumentace”Viz také:
- MVC-Pattern pro integraci ovladače
- Repository-Pattern pro přístup k datům
- DTO-Pattern pro objekty přenosu dat
- Testování pro servisní testování
Tagy: #service-layer #business-logic #dependency-injection #design-patterns