Why put a DAO layer over a persistence layer (like JDO or Hibernate)

前端 未结 10 819
梦谈多话
梦谈多话 2021-01-31 09:13

Data Access Objects (DAOs) are a common design pattern, and recommended by Sun. But the earliest examples of Java DAOs interacted directly with relational databases -- they were

相关标签:
10条回答
  • 2021-01-31 09:16

    One word: transactions

    Take the situation where I have to perform two data update operations in a single transaction. These operations together form a logical unit of work. My business logic wants to express itself in terms of that unit of work, and it doesn't want to bother itself with transaction boundaries.

    So I write a DAO. Take this pseudo code using Spring transactions and hibernate:

    edited to remove HQL that was offending @Roger so much but which wasn't relevant to the point

    @Transactional
    public void doUnitOfWork() {
      // some persistence operation here
      // some other persistence operation here
    }
    

    My business logic calls doUnitOfWork(), which begins a transaction, performs both persistence operations, and then commits. It neither knows nor cares about the transaction, or what operations are performed.

    Furthermore, if the DAO implements an interface with the doUnitOfWork() method, then the business logic can code to the interface, making it easier to unit test.

    Generally, I always wrap my data access operations in a DAO, and whack an interface around it.

    0 讨论(0)
  • 2021-01-31 09:18

    If you use an ORM: Enjoy their Transparent Persistence Support! Don't use DAOs to wrap ORM APIs. As it was well said here, DAOs are before ORMs. ORMs has introduced concepts from OODBMS, like Transparent Persistence and Persistence by Reachability. You have to take advantage of that, because it will make your life easier and your code beautiful. Suppose your are modeling departments and employees... One use case might be creating a new department, creating a new employee and adding the employee to the department... what you would do the?

    //start persistence context
    ...
    Department dept1 = new Department("department1");
    dept1.addEmployee(new Employee("José", 10503f));
    
    em.persist(dept1);
    ...
    //close persistence context
    

    Department, Employee and their relation are persistent now.

    Suppose now you have to add an existing Employee to and existing Department... what you would do? pretty simple:

    //start persistence context
    ...
    Department aDepart = hibernateSession.load(Department.class, dId);
    Employee anEmployee = hibernateSession.load(Employee.class, eId);
    
    aDepart.addEmployee(anEmployee);     
    ...
    //close persistence context
    

    Pretty simple thanks to the Transparent Persistence and Persistence by Reachability that Hibernate (like other ORMs) implements. No DAOs at all.

    Just code your domain model and think like you are persisting in memory. With a good mapping strategy, the ORM will transparently persist what do you in memory.

    More examples here: http://www.copypasteisforword.com/notes/hibernate-transparent-persistence http://www.copypasteisforword.com/notes/hibernate-transparent-persistence-ii

    0 讨论(0)
  • 2021-01-31 09:26

    You make some points. But I nevertheless use a Dao layer, here's why:

    1. Database accesses are calls to a remote system. In all such cases (also web-service, ajax etc...), the granularity of interaction need to be big enough. Many tiny calls would kill performance. This performance necessity requires often a different view of the system, or layer (here, the Dao layer).

    2. Sometimes, your persistence operation is only to load/save/delete an object. One unique Dao (or a superclass ; consider Generics) can be responsible for this, so you don't have to code these methods again and again.
      But often, you also have specific needs, like running a specific request that is not automatically created by the ORM. There, you code your specific need with a specific Dao method (reuse is often possible).
      Having regular and specific needs in the same layer allow for reuse (for example, interception can ensure that a database connection is open/commited when needed).

    0 讨论(0)
  • 2021-01-31 09:27

    DAO has lost its meaning over time.

    During the J2EE days when it became a popular pattern, a DAO was a class where you could simultaneously cater for multiple sources of data - a database by one vendor, a database by another, a file - and provide a single place to wrap queries to communicate for data.

    There was plenty of scope for reuse, so an DAO object for a particular entity may well extend an abstract DAO which housed the re-usable stuff, which in itself implemented a DAO interface.

    Post-J2EE/EJB, the DataMapper and DataSource patterns (or for simple systems, ActiveRecord) became popular to perform the same role. However, DAO became a buzzword for any object involved with persistence.

    Nowdays, the term 'DAO' has sadly become a synonym for "a class which enables me to communicate with my database".

    With ORM / JPA, much of the rationale for a true, J2EE era DAO is provided out of the box.

    In the case of a latter DataSource pattern, JPA's EntityManager is akin to the DataSource, but is usually provided via a PersistenceUnit XML definition and instantiated via IoC.

    The CRUD methods which once lived in a DAO or Mapper can now be provided exactly once using the Repository pattern. There's no need for AbstractDAO's - the ORM products are clever enough to accept an Object() and know where it is persisting it.

    0 讨论(0)
  • 2021-01-31 09:33

    I believe most DAOs are added by people for histerical (historical ;] ) reasons. You are right in that they were intially meant as a convenient encapsulation of the SQL glue required to perform the CRUD operations in pre ORM days. Nowadays, with transparent persistence, their role is now largely redundant.

    What is now appropriate is the concepts of Repositories and Services:

    Repository: A class that stores a collection of query methods implemented in ORM specific code (eg, Hibernate or JDO)

    Typically you can create an abstract base class Repository and then provide an ORM specific implementation into which you implement all the query methods in code that is specific to your ORM. The great thing about this approach is that you can create a MockRepository implemenation to help test your app without using the DB.

    Service: A class that stores a collection of methods that can orchestrate non trivial changes/additions to the object model (typically ORM independent code).

    This helps to keep your app largely ORM independent - to port the app to another ORM really only involves the implementation of a new ORM specific Repository class(es).

    0 讨论(0)
  • 2021-01-31 09:33

    It's actually must simpler than all of these answers make it out to be. These patterns are all about layers. You don't want circular references to you make layers that can only know about things above them. You want your UICode to be able to reference any and all Services, your Service code to be able to reference any and all DAOs.

    1. DAO
    2. Service
    3. UICode

    with the POJOs being passed from top to bottom.

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