“Do not use Abstract Base class in Design; but in Modeling/Analysis”

后端 未结 2 1840
无人共我
无人共我 2020-11-29 10:37

I am newbie to SOA though I have some experience in OOAD.

One of the guidelines for SOA design is “Use Abstract Classes for Modeling only. Omit them from Design”. Th

相关标签:
2条回答
  • 2020-11-29 11:00

    I would go pretty much with what others have said here, but probably needs to add these:

    • Most SOA systems use Web Services for communication. Web Services expose their interface via WSDL. WSDL does not have any understanding of inheritance.
    • All behaviour in your DTOs will be lost when they cross the wire
    • All private/protected fields will be lost when they cross the wire

    Imagine this scenario (case is silly but illustrative):

    public abstract class BankAccount
    {
        private DateTime _creationDate = DateTime.Now;
    
        public DateTime CreationDate
        {
            get { return _creationDate; }
            set { _creationDate = value; }
        }
    
        public virtual string CreationDateUniversal
        {
            get { return _creationDate.ToUniversalTime().ToString(); }
        }
    }
    
    public class SavingAccount : BankAccount
    {
        public override string CreationDateUniversal
        {
            get
            {
                return base.CreationDateUniversal + " UTC";
            }
        }
    }
    

    And now you have used "Add Service Reference" or "Add Web Reference" on your client (and not re-use of the assemblies) to access the the saving account.

    SavingAccount account = serviceProxy.GetSavingAccountById(id);
    account.CreationDate = DateTime.Now;
    var creationDateUniversal = account.CreationDateUniversal; // out of sync!!
    

    What is going to happen is the changes to the CreationDate will not be reciprocated to the CreationDateUniversal since there is no implementation crossed the wire, only the value of CreationDateUniversal at the time of serialization at the server.

    0 讨论(0)
  • 2020-11-29 11:20

    It sounds like you are trying to use SOA to remotely access your object model. You would be better of looking at the interactions and capabilities you want your service to expose and avoid exposing inheritance details of your services implementation.

    So in this instance where you need a list of user accounts your interface would look something like

    [ServiceContract]
    interface ISomeService
    {
        [OperationContract]
        Collection<AccountSummary> ListAccountsForUser(
            User user /*This information could be out of band in a claim*/);
    }
    
    [DataContract]
    class AccountSummary
    {
         [DataMember]
         public string AccountNumber {get;set;}
         [DataMember]
         public string AccountType {get;set;}
         //Other account summary information
    }
    

    if you do decide to go down the inheritance route, you can use the KnownType attribute, but be aware that this will add some type information into the message being sent across the wire which may limit your interoperability in some cases.

    Update:

    I was a bit limited for time earlier when I answered, so I'll try and elaborate on why I prefer this style.

    I would not advise exposing your OOAD via DTOs in a seperate layer this usually leads to a bloated interface where you pass around a lot of data that isn't used and religously map it into and out of what is essentially a copy of your domain model with all the logic deleted, and I just don't see the value. I usually design my service layer around the operations that it exposes and I use DTOs for the definition of the service interactions.

    Using DTOs based on exposed operations and not on the domain model helps keep the service encapsulation and reduces coupling to the domain model. By not exposing my domain model, I don't have to make any compromises on field visibility or inheritance for the sake of serialization.

    for example if I was exposing a Transfer method from one account to another the service interface would look something like this:

    [ServiceContract]
    interface ISomeService
    {
        [OperationContract]
        TransferResult Transfer(TransferRequest request);
    }
    
    [DataContract]
    class TransferRequest
    {
         [DataMember]
         public string FromAccountNumber {get;set;}
         [DataMember]
         public string ToAccountNumber {get;set;}
         [DataMember]
         public Money Amount {get;set;}
    }
    
    class SomeService : ISomeService
    {
        TransferResult Transfer(TransferRequest request)
        {
            //Check parameters...omitted for clarity
            var from = repository.Load<Account>(request.FromAccountNumber);
            //Assert that the caller is authorised to request transfer on this account
            var to = repository.Load<Account>(request.ToAccountNumber);
            from.Transfer(to, request.Amount);
            //Build an appropriate response (or fault)
        }
    }
    

    now from this interface it is very clear to the conusmer what the required data to call this operation is. If I implemented this as

    [ServiceContract]
    interface ISomeService
    {
        [OperationContract]
        TransferResult Transfer(AccountDto from, AccountDto to, MoneyDto dto);
    }
    

    and AccountDto is a copy of the fields in account, as a consumer, which fields should I populate? All of them? If a new property is added to support a new operation, all users of all operations can now see this property. WCF allows me to mark this property as non mandatory so that I don't break all of my other clients, but if it is mandatory to the new operation the client will only find out when they call the operation.

    Worse, as the service implementer, what happens if they have provided me with a current balance? should I trust it?

    The general rule here is to ask who owns the data, the client or the service? If the client owns it, then it can pass it to the service and after doing some basic checks, the service can use it. If the service owns it, the client should only pass enough information for the service to retrieve what it needs. This allows the service to maintain the consistency of the data that it owns.

    In this example, the service owns the account information and the key to locate it is an account number. While the service may validate the amount (is positive, supported currency etc.) this is owned by the client and therefore we expect all fields on the DTO to be populated.

    In summary, I have seen it done all 3 ways, but designing DTOs around specific operations has been by far the most successful both from service and consumer implementations. It allows operations to evolve independently and is very explicit about what is expected by the service and what will be returned to the client.

    0 讨论(0)
提交回复
热议问题