How to solve the violations of the Law of Demeter?

后端 未结 10 1258
一个人的身影
一个人的身影 2021-01-29 22:23

A colleague and I designed a system for our customer, and in our opinion we created a nice clean design. But I\'m having problems with some coupling we\'ve introduced. I could t

相关标签:
10条回答
  • 2021-01-29 23:14

    I was struggling with the LoD just like many of you were until I watched The Clean Code Talks" session called:

    "Don't Look For Things"

    The video helps you use Dependency Injection better, which inherently can fix the problems with LoD. By changing your design a bit, you can pass in many lower level objects or subtypes when constructing a parent object, thus preventing the parent from having to walk the dependency chain through the child objects.

    In your example, you would need to pass in AdministrationRoute to the constructor of ProtocolMedication. You'd have to redesign a few things so it made sense, but that's the idea.

    Having said that, being new to the LoD and no expert, I would tend to agree with you and DrJokepu. There are probably exceptions to the LoD like most rules, and it might not apply to your design.

    [ Being a few years late, I know this answer probably won't help the originator, but that's not why I'm posting this ]

    0 讨论(0)
  • 2021-01-29 23:16

    The traditional solution to Demeter violations is "tell, don't ask." In other words, based on your state, you should tell a managed object (any object you hold) to take some action -- and it will decide whether to do what you ask, depending on its own state.

    As a simple example: my code uses a logging framework, and I tell my logger that I want to output a debug message. The logger then decides, based on its configuration (perhaps debugging isn't enabled for it) whether or not to actually send the message to its output devices. A LoD violation in this case would be for my object to ask the logger whether it's going to do anything with the message. By doing so, I've now coupled my code to knowledge of the logger's internal state (and yes, I picked this example intentionally).

    However, the key point of this example is that the logger implements behavior.

    Where I think the LoD breaks down is when dealing with an object that represents data, with no behavior.

    In which case, IMO traversing the object graph is no different than applying an XPath expression to a DOM. And adding methods such as "isThisMedicationWarranted()" is a worse approach, because now you're distributing business rules amongst your objects, making them harder to understand.

    0 讨论(0)
  • 2021-01-29 23:20

    I'd have to assume that the business logic that requires Soluble requires other things too. If so, can some part of it be incapsulated in Medicine in a meaningful way (more meaningful than Medicine.isSoluble())?

    Another possibility (probably an overkill and not complete solution at the same time) would be to present the business rule as object of its own and use double dispatch/Visitor pattern:

    RuleCompilator
    {
      lookAt(Protocol);
      lookAt(Medicine);
      lookAt(AdminstrationProcedure) 
    }
    
    MyComplexRuleCompilator : RuleCompilator
    {
      lookaAt(Protocol)
      lookAt(AdminstrationProcedure)
    }
    
    Medicine
    {
      applyRuleCompilator(RuleCompilator c) {
        c.lookAt(this);
        AdministrationProtocol.applyRuleCompilator(c);
      }
    }
    
    0 讨论(0)
  • 2021-01-29 23:22

    I think it helps to remember the raison d'être of LoD. That is, if details change in chains of relationships, your code could break. Since the classes you have are abstractions close to the problem domain, then the relations aren't likely to change if the problem stays the same, e.g., Protocol uses Discipline to get its work done, but the abstractions are high level and not likely to change. Think of information hiding, and it's not possible for a Protocol to ignore the existence of Disciplines, right? Maybe I'm off on the domain model understanding...

    This link between Protocol and Discipline is different than "implementation" details, such as order of lists, format of data structures, etc. that could change for performance reasons, for example. It's true this is a somewhat gray area.

    I think that if you did a domain model, you'd see more coupling than what is in your C# class diagram. [Edit] I added what I suspect are relationships in your problem domain with dashed lines in the following diagram:

    UML Diagram of Domain model

    On the other hand, you could always refactor your code by applying the Tell, don't ask metaphor:

    That is, you should endeavor to tell objects what you want them to do; do not ask them questions about their state, make a decision, and then tell them what to do.

    You already refactored the first problem (BLL) with your answer. (Another way to abstract BLL further would be with a rule engine.)

    To refactor the second problem (repositories), the inner code

        p.Kind.Discipline.Id == discipline.Id
    

    could probably be replaced by some kind of .equals() call using a standard API for Collections (I'm more of a Java programmer, so I'm not sure of the precise C# equivalent). The idea is to hide the details of how to determine a match.

    To refactor the third problem (inside the UI), I'm also not familiar with ASP.NET, but if there's a way to tell a Kind object to return the names of Disciplines (rather than asking it for the details as in Kind.Discipline.Name), that's the way to go to respect LoD.

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