O documento discute Domain-Driven Design (DDD), abordando conceitos como linguagem ubíqua, modelagem orientada a domínio, blocos de construção como entidades e objetos de valor. Exemplifica como esses conceitos podem ser aplicados na prática ao modelar uma camada de domínio com classes como Product, Currency e Money.
3. Domain-Driven Design?
Domain-Driven Design é uma
abordagem para o Desenvolvimento
de Software que estabelece uma forte
ligação entre a implementação e o
modelo evolutivo dos conceitos de
negócio. Implementação X Negócios
5. Primeira resposta
“Vai escrevendo código ai,
quando precisar coloque
na camada de
apresentação, no banco
de dados nos controles,
modelos, na configuração
do webserver e não
esquece do xml e aquele
if maroto no
APPLICATION_ENV”
11. Orientação a Objetos
Quando pensamos em POO logo
pensamos em Classes, Encapsulamento,
Polimorfismo, Herança...
POO é mais que isso, a essência da
POO é alinhamento do código com o
negócio, reutilização, mínimo de
acoplamento, linguagem natural...
12. Como eu uso? De onde eu
começo?
sudo apt-get install ddd (#sqn)
Domain Driven Design
15. Linguagem Ubíqua
Definições de Ubíqua
“O termo foi usado pela primeira vez pelo cientista Norte-Americano Mark Weiser
em 1988 e publicado em 1991 no seu artigo The Computer for the 21st Century”
“termo usado para descrever a onipresença da informática no cotidiano das
pessoas”
“Computação ubíqua tem como objetivo tornar a interação homem computador
invisível, ou seja, integrar a informática com as ações e comportamentos naturais
das pessoas”
Para que um software atenda um domínio, é necessário que se estabeleça, em
primeiro lugar, uma linguagem Ubíquia ou seja, linguagem comum, com termos
bem definidos, que fazem parte do domínio do negócio e que são usados por
todas as pessoas que fazem parte do processo de desenvolvimento.
16. Projeto Dirigido pelo Modelo (Model Driven Design – MDD). A ideia por
trás de MDD é a de que o seu modelo abstrato deve ser uma representação
perfeita do seu domínio.
MDD – Model Driven Design
UML ajuda muito nessa fase mas, use com cuidado!
19. DDD na prática
Camada de domínio. Para modelar essa parte, utilizamos alguns Padrões propostos em DDD.
Chamados de blocos de construção, utilizados para representar nosso modelo abstrato. Esses blocos
podem ser:
Entidades - classes de objetos que necessitam de uma identidade.
Objetos de Valores - objetos que só carregam valores, mas que não possuem distinção
de identidade.
Agregados - compostos de Entidades ou Objetos de Valores que são encapsulados numa
única classe
Fábricas - classes responsáveis pelo processo de criação dos Agregados ou dos Objetos
de Valores
Serviços - classes que contêm lógica de negócio, mas que não pertencem a nenhuma
Entidade ou Objetos de Valores.
Repositórios - classes responsáveis por administrar o ciclo de vida dos outros objetos,
normalmente Entidades, Objetos de Valor e Agregados.
20. DDD na prática
Camada de domínio. Para modelar essa parte, utilizamos alguns Padrões propostos em DDD.
Chamados de blocos de construção, utilizados para representar nosso modelo abstrato. Esses blocos
podem ser:
Entidades - classes de objetos que necessitam de uma identidade.
Objetos de Valores - objetos que só carregam valores, mas que não possuem distinção
de identidade.
Agregados - compostos de Entidades ou Objetos de Valores que são encapsulados numa
única classe
Fábricas - classes responsáveis pelo processo de criação dos Agregados ou dos Objetos
de Valores
Serviços - classes que contêm lógica de negócio, mas que não pertencem a nenhuma
Entidade ou Objetos de Valores.
Repositórios - classes responsáveis por administrar o ciclo de vida dos outros objetos,
normalmente Entidades, Objetos de Valor e Agregados.
21. Objeto de Valor (Value Object)
Muitos objetos não possuem nenhuma identidade conceitual. Esses objetos descrevem alguma
característica de alguma coisa.
22. Características
- Medir
- Quantificar
- Descrever
Imutabilidade
“This is one of the most important aspects of a Value Object to grasp. Object values should not be
able to be altered over their life-time. Because of this immutability, Value Objects are easy to reason,
test and are free of undesired/unexpected side-effects.” (Carlos Buenosvinos, Christian Soronellas
and Keyvan Akbary DDD in PHP)
24. class Currency
{
private $isoCode;
public function __construct($anIsoCode)
{
$this->setIsoCode($anIsoCode);
}
private function setIsoCode($anIsoCode)
{
if (!preg_match('/^[A-Z]{3}$/', $anIsoCode)) {
throw new InvalidArgumentException();
}
$this->isoCode = $anIsoCode;
}
public function isoCode()
{
return $this->isoCode;
}
}
25. class Money
{
private $amount;
private $currency;
public function __construct($anAmount, Currency $aCurrency)
{
$this->setAmount($anAmount);
$this->setCurrency($aCurrency);
}
private function setAmount($anAmount)
{
$this->amount = (int) $anAmount;
}
private function setCurrency(Currency $aCurrency)
{
$this->currency = $aCurrency;
}
}
26. class Money
{
/……/
public static function fromMoney(Money $aMoney)
{
return new self(
$aMoney->amount(),
$aMoney->currency()
);
}
public static function ofCurrency(Currency $aCurrency)
{
return new self(0, $aCurrency);
}
}
27. class Money
{
/……/
public function increaseAmountBy($anAmount)
{
return new self(
$this->amount() + $anAmount,
$this->currency()
);
}
public function add(Money $money)
{
if (!$money->currency()->equals($this->currency())) {
throw new InvalidArgumentException();
}
return new self(
$money->amount() + $this->amount(),
$this->currency()
);
}
}
29. class MoneyTest extends PHPUnit_Test_TestCase
{
Const BRAZIL_CURRENCY = "BRL";
public function testCopiedMoneyShouldRepresentSameValue()
{
$aMoney = new Money(100, new Currency(self::BRAZIL_CURRENCY));
$copiedMoney = Money::fromMoney($aMoney);
$this->assertTrue($aMoney->equals($copiedMoney));
}
public function testOriginalMoneyShouldNotBeModifiedOnAddition()
{
$aMoney = new Money(100, new Currency(self::BRAZIL_CURRENCY));
$aMoney->add(new Money(20, new Currency(self::BRAZIL_CURRENCY)));
$this->assertEquals(100, $aMoney->amount());
}
public function testMoneysShouldBeAdded()
{
$aMoney = new Money(100, new Currency(self::BRAZIL_CURRENCY));
$newMoney = $aMoney->add(new Money(20, new Currency(self::BRAZIL_CURRENCY)));
$this->assertEquals(120, $newMoney->amount());
}
}
30. Persistindo Objetos de Valor
Objetos de valor não são persistidos por conta própria, até por que eles não
possuem identificação. Geralmente um Objeto de Valor é gravado como atributos
de um agregado que possue identificação lógica no sistema.
31. <?php
class Product
{
private $productId;
private $name;
private $price;
public function __construct(
$aProductId,
$aName,
Money $aPrice
) {
$this->setProductId($aProductId);
$this->setName($aName);
$this->setPrice($aPrice);
}
// ...
}
34. class DbalProductRepository extends DbalRepository implements
ProductRepository
{
public function add(Product $aProduct)
{
$sql = 'INSERT INTO products VALUES (?, ?, ?, ?)';
$stmt = $this->connection()->prepare($sql);
$stmt->bindValue(1, $aProduct->id());
$stmt->bindValue(2, $aProduct->name());
$stmt->bindValue(3, $aProduct->price()->amount());
$stmt->bindValue(4, $aProduct->price()->currency()->isoCode());
$stmt->execute();
// ...
}
}
35. class DbalProductRepository extends DbalRepository implements
ProductRepository
{
public function productOfId($anId)
{
$sql = 'SELECT * FROM products WHERE id = ?';
$stmt = $this->connection()->prepare($sql);
$stmt->bindValue(1, $anId);
$res = $stmt->execute();
// ...
return new Product(
$row['id'],
$row['name'],
new Money(
$row['price_amount'],
new Currency(
$row['price_currency']
)
)
);
}
}
36. Entidade (Entity)
Muitos objetos não são fundamentalmente definidos por seus atributos, mas sim por uma linha de
continuidade e identidade
37. class Person {
private $identificationNumber;
private $firstName;
private $lastName;
public function __construct($anIdentificationNumber, $aFirstName, $aLastName) {
$this->identificationNumber = $anIdentificationNumber;
$this->firstName = $aFirstName;
$this->lastName = $aLastName;
}
public function identificationNumber() {
return $this->identificationNumber;
}
public function firstName() {
return $this->firstName;
}
public function lastName() {
return $this->lastName;
}
}
43. Referência
Domain-Driven Design in PHP
Real examples written in PHP showcasing DDD Architectural
Styles, Tactical Design, and Bounded Context Integration.
Carlos Buenosvinos, Christian Soronellas and Keyvan Akbary
http://leanpub.com/ddd-in-php