1

我有以下域实体:

public class CartItem
{
    public virtual Guid Id { get; set; }
    public virtual Guid SessionId  { get; set; }
    public virtual int Quantity  { get; set; }
    public virtual Product Product  { get; set; }
}

我有以下 DTO:

public class CartItemDTO
{
    public CartItemDTO(CartItem cartItem)
    {
        Id = cartItem.Id;
        Quantity = cartItem.Quantity;
        Name = cartItem.Product.Name;
        Price = cartItem.Product.Price;
    }

    public Guid Id { get; private set; }
    public int Quantity { get; private set; }
    public string Name { get; private set; }
    public decimal Price { get; private set; }
}

当前的工作流程非常简单:我的存储库返回一个 CartItem 类型的 IEnumerable。我的服务将其转换为 dto (CartItemDTO)。然后我的控制器将其传递给视图。到目前为止,一切都很好。

现在我想为每个行项目实施总计。我将以下属性添加到 CartItemDTO。

public decimal Total { get; private set; }

然后,我将以下代码添加到 CartItemDTO 构造函数。

Total = cartItem.Quantity * cartItem.Product.Price;

我的第一个问题是这是否是最佳实践方法?如果不是,为什么?我应该在其他地方添加 Total 属性吗?如果是这样,为什么?

我还想为整个购物车实现总计,因此我创建了一个新类(如下)并修改了我的服务以返回它。

public class CartItemResult
{
    public CartItemResult(IEnumerable<CartItemDTO> result)
    {
        CartItems = new List<CartItemDTO>(result);
        Total = result.Sum(total => total.Total);
    }

    public IList<CartItemDTO> CartItems { get; private set; }
    public decimal Total { get; private set; }
}

我现在可以将新类传递给视图,或者创建一个单独的 ViewModel 并将新类的内容传递给 ViewModel 并将其传递给视图。

我的第二个问题是,这种方法是否是最佳实践方法?如果不是,为什么以及我应该做些什么不同的事情?

4

2 回答 2

5

DTO 是用于传输数据的纯类。它不应该包含任何逻辑。您已经添加了域驱动设计标签,所以我认为您想要使用域对象和域服务。您的域对象 (CartItem) 负责计算项目的总数。域对象是数据+业务逻辑和使用该数据的规则。计算总数是业务逻辑。同样对于您的第二部分(CartItemResult),这通常由称为域服务的东西处理。域服务提供与单个域对象无关的业务逻辑,而是与许多域对象一起工作。

但是您还应该考虑项目的复杂性。你真的需要这个吗?不要仅仅因为它们存在就使用它们,总是找到使用它们的理由。David Neale 提到的方法对于某些应用程序也是正确的。

最好的问候, 拉迪斯拉夫

于 2010-08-09T21:56:10.937 回答
3

就个人而言,对于数据检索,我建议从存储库中返回 POCO 实体并将其映射到控制器(Automapper)中的 ViewModel 。真的没有必要有一个中介对象。

Total应该在哪里实现属性取决于您的设计。该属性是否在域中用于任何业务逻辑?可以吗?如果不是,那么您可能会认为它只是一个演示问题并且适用于 ViewModel。

对于数据更新 -CartItemUpdateViewModel在您的 UI 中返回类似 a 的内容,它映射到控制器中的域实体,CartItemUpdateDetails然后将其传递给存储库方法。

至于你的问题CartItemResult。我个人会将Total逻辑放在 ViewModel 中,并从返回的存储库方法中将该 ViewModel 填充到控制器中IEnumerable<CartItem>

看看WhoCanHelpMe项目,这是一个展示 DDD 的优秀企业应用程序。

于 2010-08-09T11:31:19.590 回答