问题
数据传输对象(DTO)可以引用域模型的值对象(VO)吗?
上下文
在我的域中,我有一个进口商,它从集合中导入集合。收集由进口商依赖的收集器由DTO构建。既然导入器和收集器都是我域的服务(接口),那么DTO可以引用域值对象,还是仅在处理集合(聚合的生成)时才坚持使用原语并将它们变成值对象?
收集器实现,其中将构建由域模型中的值对象构成的DTO
<?php
/**
* i-MSCP Patcher plugin
*
* @author Laurent Declercq <l.declercq@nuxwin.com>
* @copyright (C) 2019 Laurent Declercq <l.declercq@nuxwin.com>
* @license i-MSCP License <https://www.i-mscp.net/license-agreement.html>
*/
/**
* @noinspection
* PhpUnhandledExceptionInspection
* PhpDocMissingThrowsInspection
*/
declare(strict_types=1);
namespace iMSCP\Plugin\Patcher\Infrastructure\Domain\Service\Component\Importer;
use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentBuild;
use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentName;
use iMSCP\Plugin\Patcher\Domain\Model\Component\ComponentVersion;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\ComponentCollector;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\DTO\ComponentDTO;
use iMSCP\Plugin\Patcher\Domain\Service\Component\Importer\DTO\ComponentDTOCollection;
use iMSCP_Config_Handler_File as MergedConfig;
use iMSCP_Plugin_Manager as PluginManager;
use RuntimeException;
use Throwable;
/**
* Class DefaultComponentCollector
* @package iMSCP\Plugin\Patcher\Infrastructure\Domain\Service\Component\Importer
*/
class DefaultComponentCollector implements ComponentCollector
{
/**
* @var MergedConfig
*/
private $mergedConfig;
/**
* @var PluginManager
*/
private $pluginManager;
/**
* DefaultComponentCollector constructor.
*
* @param MergedConfig $mergedConfig
* @param PluginManager $pluginManager
*/
public function __construct(
MergedConfig $mergedConfig, PluginManager $pluginManager
)
{
$this->mergedConfig = $mergedConfig;
$this->pluginManager = $pluginManager;
}
/**
* @inheritDoc
*/
public function collect(ComponentDTOCollection $collection): void
{
try {
// Core
$collection->add(new ComponentDTO(
ComponentName::fromString('core'),
ComponentVersion::fromString($this->mergedConfig['Version']),
ComponentBuild::fromString($this->mergedConfig['Build'])
));
// Plugins
$this->collectComponentsFromCorePluginManager($collection);
} catch (Throwable $e) {
throw new RuntimeException(sprintf(
"Couldn't collect list of components: %s", $e->getMessage()
), $e->getCode(), $e);
}
}
/**
* Collects components from the i-MSCP core plugin manager.
*
* @param ComponentDTOCollection $collection
* @return void
*/
private function collectComponentsFromCorePluginManager(
ComponentDTOCollection $collection
): void
{
$pluginList = $this->pluginManager->pluginGetList('all', false);
foreach ($pluginList as $pluginName) {
$pluginInfo = $this->pluginManager
->pluginGetInfo($pluginName)
->toArray();
$componentDTO = new ComponentDTO(
ComponentName::fromString($pluginInfo['name']),
ComponentVersion::fromString($pluginInfo['version']),
ComponentBuild::fromString((string)$pluginInfo['build'])
);
$collection->add($componentDTO);
}
}
}
答案 0 :(得分:4)
数据传输对象(DTO)可以引用域模型的值对象(VO)吗?
是的,但是您要对此非常小心。
data transfer object的核心是消息。为了使消息达到其目的,发送者和接收者都必须对其语义具有兼容的理解。对DTO模式进行不兼容的更改需要对接收方进行相应的更改。
在域模型中,值对象不是消息。它是结构化的信息,纯粹是当前模型的实现细节。如果我们要部署模型的新版本,该版本使用完全不同的值排列或其底层数据结构,则可以。
因此,使DTO(应该是稳定的)依赖于值对象(不能保证稳定)会为将来的问题创造机会。
如果您的价值观很稳定,那么风险就会降低。