views:

341

answers:

6

In our legacy JEE application, there are loads of value object (VO) classes which typically contain only getters and setters, maybe equals() and hashCode(). These are (typically) the entities to be saved in persistence storage. (For the record, our app has no EJBs - although that might change in the future -, and we use Hibernate for persisting our entities.) All the business logic to manipulate the data in VOs is in separate classes (not EJBs, just POJOs). My OO mindset hates this, as I do believe that the operations on a given class should reside in that same class. So I have an urge to refactor to move logic into the related VOs.

I just had a discussion with a co-worker who is much more experienced in JEE than me, and he confirmed that dumb entities at least used to be the recommended way to go. However, he has also read opinions recently which question the validity of this stance.

I understand that there are issues which at least limit what can be put inside an entity class:

  • it should not have direct dependency to the data layer (e.g. query code should rather go into separate DAOs)
  • if it is directly exposed to higher layers or to the client (e.g. via SOAP), its interface may need to be limited

Are there any more valid reasons not to move logic into my entities? Or any other concerns to take into account?

+6  A: 

I think your point is valid.
See this for more - http://martinfowler.com/bliki/AnemicDomainModel.html
With JPA, the Entities are light-weight objects. So I don't think its any problem having logic in them.

In case of use with SOAP/Web Services, I'd add a separate Facade Layer.

Padmarag
Agreed. Using objects with only getters and setters goes completely against OO and is more like using structs in C. I like C, but this is Java and I cringe everytime I have to wade trhough a class containing nothing else but boiler-plate code in the form of getX() and setX()
Martin Wickman
@Padmarag Excellent article, thanks :-)
Péter Török
I wish it was that simple :)
ewernli
Interesting article but I'm not sure stating the anemic domain model is an anti-pattern is entirely fair. In fact, it brings about the question of what is a pattern in the first place.GOF patterns make sense from a logical point of view. So can MVC be qualified as a pattern in the same sense of the word?
James P.
@James P. Why couldn't it? I don't see your point.
Péter Török
Well, GOF patterns are not only elegant but also answer a series of problems that a programmer will encounter sooner or later. MVC is more conceptual in nature as are application layers, the latter being more often referred to as an architecture.To get to the point, nothing really justifies anemic. It's just that an overly rich domain would undermine the role of the Service layer. If you couple that with the debate on whether DAOs should exist with ORM the question is whether layers should exist at all. P.S:I am for neither and wouldn't mind learning from someone who prefers rich domain :)
James P.
@James P. IMO MVC as well solves a concrete problem. You can argue whether it's a design or an architectural pattern - to be frank, I don't care much. Layers also are useful in some situations, superfluous in others. I feel that you are trying to pull the conversation towards a more general / philosophical direction, and I prefer to stay on the ground and to focus on my actual problem in its concrete context.
Péter Török
Ok. Tell you what, I'll ask a contact about any possible pros and cons.
James P.
I think the issue with calling MVC a pattern is that it is really a combination of 3 GoF design patterns: Observer, Composite, and Strategy. The conjunction of these three patterns forms an "architectual" design pattern.
Matthew Flynn
A: 

The convention you are referring to is to adopt an anemic domain model, as opposed to rich domain model, where Entities are simple POJOs annotated as beans (@Entity) with just the bare minimum in terms of getters and setters. So a class with a toXML() method would be considered as rich domain. I guess it helps to keep a clear view of what is being mapped to a relational database even if granularity differs.

The advantage is that there is a clear separation between logic and data (this is where the OO philosophy is broken). The approach is to group these into business logic classes or Services. This is according to a layered architecture with respective layers being Domain, Services, DAO and UI.

That's the theory.

Edit: Just to clarify, when I say there's a clear separation between logic and data, I mean that one object is data orientated and another is method orientated in a way that could be considered as a lapse back to procedural.

James P.
@James P. I am afraid you don't quite understand the point of OO if you consider it "broken" because of this :-(
Péter Török
It's possible and I'm always open to enlightenment ;) . Correct me if I'm wrong but it sounds like the encapsulation principle is at the center of the debate here.
James P.
@James P. I am not sure I understand where you are aiming to. Anyway, I would not like to "debate" about OO because a) I would prefer to keep this topic on focus, b) opinions about OO are (rightly) subjective and context-dependant, c) 600 chars would not be enough for this anyway.
Péter Török
I've posted a second answer with a summary of some feedback that a contact has given me on the question.
James P.
+2  A: 

The DTO and VO are supposed to be used to transfer data and don't embed logic. The business objects on the other hand are supposed to embed some logic. I say some, because there is always a balance to find between what you put in services which coordinate logic involving several business objects and what you put in the business objects themselves. Typical logic in the business objects can be validation, field computation, or other operation that impact only one business object at a time.

Note that I haven't mentioned the term entity so far. Persistent entities were popularized with ORM and we nowadays try to use persistent entities both as DTO and business object at the same time. That is, the entity themselves flow between layers and tiers, and contain some logic.

Are there any more valid reasons not to move logic into my entities? Or any other concerns to take into account?

As you pointed out, it's all a matter of dependencies and what you expose. As long as the entities are dumb (close to DTO) they can be isolated in a dedicated jar easily that serves as API of the layer. The more logic you put in the entities, the harder it becomes to do that. Pay attention to what you expose and what you depend on (the load the class, the client will need to have the depend class as well). This applies to exceptions, inheritance hierarchy, etc.

Just to give an example, I had a project where the entities had a method toXml(...) used in the business layer. As a consequence, client of the entities depended on XML.

But if you don't care too much about layers, and strict separation between API and implementation, I think it's good to move some logic in the entities.

EDIT

This question has been discussed many time and will probably continue to be discussed as there is no definitive answer. A few interesting links:

ewernli
I found most answers useful and it was difficult to pick one as "the best". However, I found yours the most balanced and insightful, so you get the check mark :-) Thanks.
Péter Török
+1  A: 

The main problem with adding logic to those classes is, they would need more attributes to keep track the object state, and these extra attributes generally do not need to be serialized. This means extra work is needed in the serialization mechanism of those classes.

Considering that many projects have a mixture of jr. programmers and sr. programmers and most of the work is performed by the jr's who don't understand ( or care ) about optimal serialization, it is much more easy to have this plain old java objects as "value objects" which pretty much just pass and receive data and put the logic in other place.

If you manage to have an architecture where the logic is placed in a business object ( that is VO + Logic ) I think that would be better also. Just bear in mind the whole team is in the same page and they don't duplicate code and logic ( which never happens right? )

Short answer: No, they should not be dumb always, but is a much easier to have them that way.

OscarRyz
If serialization is an issue, you can make the additional attributes transient. Not a big deal.
Matthew Flynn
@Matthew of course, as long as newbies understand what are you talking about.
OscarRyz
+2  A: 

Beyond the Fowler article mentioned earlier, there is a full treatise on rich vs. anemic domain models in Eric Evans's book Domain Driven Design(2004).

Also, check out http://domaindrivendesign.org/

Matthew Flynn
+2  A: 

Right, here's a summary of the feedback I got from my Java EE trainer.

From a pratical point of view, striking a compromise between anemic and rich domain by moving logic shouldn't a problem as long as the methods work with the attributes of the Entity. When it comes to a rich domain, the line has to be drawn somewhere and apparently both Fowler and King have emitted comments in that direction.

For example, consider a calculateInterestRate() method inside a BankAccount that fetches information from other domain objects like verifying how long someone has been a client. To avoid dependency, one could split the method accross objects but this approach means that code can end being strewn accross several classes. At this point, one might as well make a InterestCalculator class.

Another thing to take into account is thread safety. Singleton DAOs and Services handled by Spring should be thread safe whereas anything in the domain model will be exposed to concurrency issues.

Lastly, there's the problem of maintenance. Are you sure you will be around to maintain the application in a couple of year's time? The choices you make may appear to be justified but are you certain that the next developper will have the expertise necessary to easily understand your code?

James P.