8

背景: 我有一个 Person 域对象。它是一个聚合根。我已经包括了下面课程的一部分。

我正在公开执行对象行为的方法。例如,要添加 BankAccount,我有 AddBankAccount() 方法。我没有包含该类的所有方法,但足以说明任何公共属性都必须使用方法进行更新。

我将创建一个 IPerson 存储库来处理 CRUD 操作。

public interface IPersonRepository
{ 
    void Save(Person p);
    //...other methods
}

问题:当我们更新现有人员时,我如何告诉存储库哪些字段需要更新?例如,如果我向现有人员添加银行帐户,当调用 repository.Save() 时,如何将此信息传达给存储库?

在存储库中,很容易确定何时创建新人员,但是当现有人员存在并且您更新该人的字段时,我不确定如何将其传达给存储库。

我不想用更新哪些字段的信息来污染我的 Person 对象。

我可以在存储库上使用单独的方法,例如 .UpdateEmail()、AddBankAccount(),但这感觉有点矫枉过正。我想在存储库上有一个简单的 .Save() 方法,它确定需要以某种方式更新的内容。

其他人是如何处理这种情况的?

我搜索了网络和stackoverflow,但没有找到任何东西。我一定没有正确搜索,因为在 DDD 范式中的持久性方面这似乎很简单。我对 DDD 的理解也可能离我很远 :-)

public class Person : DomainObject
{
    public Person(int Id, string FirstName, string LastName,
        string Name, string Email)
    {
        this.Id = Id;
        this.CreditCards = new List<CreditCard>();
        this.BankAccounts = new List<BankAccount>();
        this.PhoneNumbers = new List<PhoneNumber>();
        this.Sponsorships = new List<Sponsorship>();
    }

    public string FirstName { get; private set; }
    public string LastName { get; private set; }
    public string Name{ get; private set; }
    public string Email { get; private set; }
    public string LoginName { get; private set; }

    public ICollection<CreditCard> CreditCards { get; private set; }

    public ICollection<BankAccount> BankAccounts { get; private set; }

    public ICollection<PhoneNumber> PhoneNumbers { get; private set; }  

    public void AddBankAccount(BankAccount accountToAdd, IBankAccountValidator bankAccountValidator)
    {
        bankAccountValidator.Validate(accountToAdd);

        this.BankAccounts.Add(accountToAdd);
    }

    public void AddCreditCard(CreditCard creditCardToAdd, ICreditCardValidator ccValidator)
    {
        ccValidator.Validate(creditCardToAdd);

        this.CreditCards.Add(creditCardToAdd);
    }

    public void UpdateEmail(string NewEmail)
    {
        this.Email = NewEmail;
    }
4

2 回答 2

3

有一个来自S#arp Architecture项目的 Repository 接口示例。它类似于PoEAA 数据映射器,因为它也用于 CRUD 操作。

public interface IRepositoryWithTypedId<T, IdT>
{
    T Get(IdT id);    
    IList<T> GetAll();    
    IList<T> FindAll(IDictionary<string, object> propertyValuePairs);    
    T FindOne(IDictionary<string, object> propertyValuePairs);    
    T SaveOrUpdate(T entity);    
    void Delete(T entity);    
    IDbContext DbContext { get; }
}

如您所见,实体的特定属性没有更新方法。整个实体作为参数提供给方法SaveOrUpdate

当您的域实体的属性被更新时,您应该告诉您的工作单元该实体是“脏的”并且应该保存到存储中(例如数据库)

PoEAA 工作单元

您不应使用有关更新字段的信息污染您的 Person 对象,但如果实体更新,则需要跟踪信息。

DomainObject如果实体是“新”、“脏”或“已删除”,则可能有类的方法告诉“工作单元”。然后您的 UoW 本身可能会调用适当的存储库方法 - “SaveOrUpdate”或“Delete”。

尽管NHibernateEntityFramework等现代 ORM 框架有自己的“工作单元”实现,但人们倾向于为它们编写自己的包装器/抽象。

于 2013-10-16T18:08:33.850 回答
0

为了解决这个问题,我正在做的是向我的域对象添加一个接口:

interface IDirtyTracker {
    bool IsDirty {get;}
    void MarkClean();
    void MarkDirty();
}

DomainObject类可以实现IDirtyTracker,然后存储库等可以IsDirty用来检查它是脏的还是干净的。

在每个进行更改的 setter 中:

void SetValue() {
    this._value = newValue;
    this.MarkDirty();
}

这不会为您提供细粒度检查,但它是一种避免在存储库级别进行一些不必要更新的简单方法。

为了使这更容易一点,GetPropertiesToIncludeInDirtyCheck可以添加一个方法,该方法将检索需要检查的属性列表。

interface IDirtyTracker {
    IENumerable<Object> GetPropertiesToIncludeInDirtyCheck();
}
于 2016-09-29T22:10:01.817 回答