Hibernate与JDO肩并肩

时间:2022-03-28 17:37:06

在theServerSide.com上有一些人声称JDO的开发人员已经把JDO带向错误的道路,并且指出Hibernate相对更优秀。然而,根据我的经验Hibernate和JDO都是O/R Mapping的优秀技术。

它们有许多共同的特征,包括:

  • 支持Plain Old Java Object (POJO) 的几近透明的持久层
  • 基於XML的object/relational映射
  • 两者都有一个"EntityManager" API - Hibernate Session 和 JDO PersistenceManager
  • 都可以在或不在Container中运行应用程序
  • 事务级和应用级的cache
  • 丰富的查询语言 (query language)
  • 能够积极和消极两种方式装载有关联的对象
  • 有效率地处理大数据集合

因此,JDO 和 Hibernate 两种版本的同一应用程序经常很相似。

关于如何装载对象和执行查询,我们来看看下面的例子。这里有两个版本的RestaurantRepository类,一个是JDO的,另一个是Hibernate的。 RestaurantRepository类定义了寻找饭店的方法:

  • findRestaurant() - 找单一饭店 (类似于 findByPrimaryKey()).
  • findAvailableRestaurants() - 执行查询,找到在特定时间特定区域营业的饭店

列表 1 显示了JDO版的 RestaurantRepository 类, 列表 2 显示了Hibernate 的版本。

列表 1 - JDO 版

public class JDORestaurantRepositoryImpl
    implements RestaurantRepositoryImpl {

  public Restaurant findRestaurant(String restaurantId) {
   PersistenceManager pm = PMRegistry.getPM();
   return (Restaurant) pm.getObjectById(
    pm.newObjectIdInstance(
     JDORestaurant.class,
     restaurantId),
    true);
  }


    private static final String QUERY_STRING =
        "serviceArea.contains(zipCode) && timeRanges.contains(tr) && "
            + "(tr.dayOfWeek == day && "
            + "(tr.openHour < hour || (tr.openHour == hour && tr.openMinute <= minute)) && "
            + "(tr.closeHour > hour || (tr.closeHour == hour && tr.closeMinute > minute)))";

    public Collection findRestaurants(
        Address deliveryAddress,
        Date deliveryTime) {
        Calendar c = Calendar.getInstance();
        c.setTime(deliveryTime);
        int dayOfWeek = c.get(Calendar.DAY_OF_WEEK);
        int hour = c.get(Calendar.HOUR_OF_DAY);
        int minute = c.get(Calendar.MINUTE);

        PersistenceManager pm = PMRegistry.getPM();
        Query query = pm.newQuery(JDORestaurant.class, QUERY_STRING);
        query.declareVariables("JDOTimeRange tr");
        query.declareParameters(
            "String zipCode, int day, int hour, int minute");
        Collection result =
            (Collection) query.executeWithArray(
                new Object[] {
                    deliveryAddress.getZip(),
                    new Integer(dayOfWeek),
                    new Integer(hour),
                    new Integer(minute)});
        return result;
    }
}

 

列表 2 - Hibernate 版

public class HibernateRestaurantRepositoryImpl
    implements RestaurantRepositoryImpl {

    public Restaurant findRestaurant(String restaurantId) {
        try {
            Session session = HibernateSessionRegistry.getSession();
            Restaurant restaurant =
                (Restaurant) session.load(
                    HibernateRestaurant.class,
                    new Long(restaurantId));
            return restaurant;
        } catch (HibernateException e) {
            throw new ApplicationRuntimeException(e);
        }
    }

    public Collection findRestaurants(
        Address deliveryAddress,
        Date deliveryTime) {
        try {
            Query query =
                makeFindRestaurantsQuery(
                    deliveryAddress,
                    deliveryTime);
            return query.list();
        } catch (HibernateException e) {
            throw new ApplicationRuntimeException(e);
        }
    }

    private Query makeFindRestaurantsQuery(
        Address deliveryAddress,
        Date deliveryTime)
        throws HibernateException {
        Calendar c = Calendar.getInstance();
        c.setTime(deliveryTime);
        int dayOfWeek = c.get(Calendar.DAY_OF_WEEK);
        int hour = c.get(Calendar.HOUR_OF_DAY);
        int minute = c.get(Calendar.MINUTE);
        Session session = HibernateSessionRegistry.getSession();
        Query query =
            session.getNamedQuery("findAvailableRestaurants");
        query.setString("zipCode", deliveryAddress.getZip());
        query.setInteger("dayOfWeek", dayOfWeek);
        query.setInteger("hour", hour);
        query.setInteger("minute", minute);
        query.setCacheable(true);
        return query;
    }
}

每个 repository 由两个方法组成。这两个方法调用相应的 persistence framework API:

  • findRestaurant() - JDO 版调用 PersistenceManager.getObjectById() 而 Hibernate版调用 Session.load()
  • findAvailableRestaurants() - 两个版本都使用Query interface来执行有名参数的查询. 有个不同的地方,Hibernate 版用 named query, named query会在一个映射文件中定义,而 JDO 版用镶嵌在类里的 query 

两个类通过 ThreadLocal-based registry 得到 Hibernate Session 和 JDO PersistenceManager。如你所见,除了方法和类的名字不同,其它code都非常相似。