工作三年了,想把自己接触过的,以及将要接触的一些知识技能做些整理。因此,开始着手长期开发一个纯技术性的生态系统——j2eelib,顾名思义,里面的各模块,和j2ee相关。

    今天要说的是其中和MVC相关的部分——自己设计的一个DAO模式,用于数据库访问层。

    首先,关于dao层相关的包结构,如下图所示。

   

java 注入的类报空指针_List

    该包结构可以看出,这套DAO模式存在两个体系。其一是application包下的应用级体系,其二是framework包下的框架级体系。

    一般而言,对于一个需要进行数据库访问的dao,比如截图中对user实体进行操作的dao,笔者认为存在这样一种继承体系思考原则:对于上层的bo层或service层,需要注入相应的userDAO属性用于数据库访问。而userDAO可以是hibernate实现,也可以是jdbc实现,或者是ibatis实现。但各种实现,都需要有一个通用的接口,使得bo层userBO能够无差别的去调用。因此在application包下存在一个IUserDAO接口,而下面的UserHibernateDAO和UserJdbcDAO分别是其下两种实现。可在属性注入时注入到bo中。   

    IUserDAO的代码如下所示:

   

package com.wantall.application.mvc.dao.interfaces;

import com.wantall.application.mvc.po.User;
import com.wantall.framework.mvc.dao.interfaces.IBaseDAO;

public interface IUserDAO extends IBaseDAO<User> {

    //void test(User user);
    
}

    作为一个简单的application应用,IUserDAO没有太多的业务方法,而仅仅是继承了另一个framework包下的IBaseDAO,这个类的存在是为了给所有的DAO业务接口提供一个基本的操作规范,其代码如下所示:

 

package com.wantall.framework.mvc.dao.interfaces;

import java.io.Serializable;
import java.util.List;

public interface IBaseDAO<T> {
    
    public int insert(T obj);
    
    public int delete(Serializable id);
    
    public int delete(T obj);
    
    public int update(T obj);
    
    public T queryOneByID(Serializable id);
    
    public List<T> queryAllForList();

}

 

    下面从具体技术实现角度继续说明。

    对于IUserDAO的实现,如UserHibernateDAO,代码如下所示:

 

package com.wantall.application.mvc.dao;

import java.io.Serializable;
import java.util.List;

import com.wantall.application.mvc.dao.interfaces.IUserDAO;
import com.wantall.application.mvc.po.User;
import com.wantall.framework.mvc.daosupport.hibernate.interfaces.IHibernateDAOSupport;

public class UserHibernateDAO implements IUserDAO {

    private IHibernateDAOSupport<User> hibernateDAOSupport;
    
    public int delete(Serializable id) {
        User user = new User();
        hibernateDAOSupport.hibDelete(user);
        return 0;
    }
    
    public int delete(User user) {
        hibernateDAOSupport.hibDelete(user);
        return 0;
    }

    public int insert(User user) {
        hibernateDAOSupport.hibInsert(user);
        return 0;
    }

    public List<User> queryAllForList() {
        return hibernateDAOSupport.hibQueryAllForList(User.class);
    }

    public User queryOneByID(Serializable id) {
        return hibernateDAOSupport.hibQuery(User.class, id);
    }

    public int update(User user) {
        hibernateDAOSupport.hibUpdate(user);
        return 0;
    }

    public void test(User user) {
        //测试脱管后无缓存
        insert(user);
        //super.getSessionFactory().evict(user.getClass());
        //hibernateDAOSupport.getSessionFactory().getCurrentSession().evict(user);
        queryOneByID(user.getId());
    }

}

    对此需要说明的是,其中存在一个hibernateDAOSupport属性。该属性告诉我们,该类是一个基于hibernate的数据库访问类。而hibernateDAOSupport属性是一个support层级的接口。笔者考虑到hibernate的实现有多种,可以是原生hibernate的操作,也可以是基于spring中的hibernateDAOSupport实现,也可以是spring中的hibernateTemplate实现,因此这里通过hibernateDAOSupport接口,屏蔽了实现细节的多样性。其代码如下:

package com.wantall.framework.mvc.daosupport.hibernate.interfaces;

import java.io.Serializable;
import java.util.List;

public interface IHibernateDAOSupport<T> {

    public void hibInsert(T obj);

    /**
     * 删除
     * 该方法必须要先查出T obj
     * @param id
     */
    public void hibDelete(Class<T> cl, Serializable id);

    /**
     * 删除
     * 该方法可以删除一个new出来的对象 无需先查出
     * @param obj
     */
    public void hibDelete(T obj);

    public void hibUpdate(T obj);

    public int excuteUpdateHQL(String hql, Object[] params);

    public T hibQuery(Class<T> cl, Serializable id);

    public List<T> hibQueryAllForList(Class<T> cl);

    @SuppressWarnings("unchecked")
    public List<T> executeListQuery(String hql, Object... Params);

    /**
     * 简单hql查询,获取出的不一定是对象,可能是属性集合
     * @param hql
     * @return
     */
    @SuppressWarnings("unchecked")
    public List<T> hibCommonQuery(String hql);

}

而为了说明这种support的继承体系,以其实现类HibernateDAOSupport的代码进行说明:

 

package com.wantall.framework.mvc.daosupport.hibernate;

import java.io.Serializable;
import java.util.List;

import org.hibernate.Query;
import org.hibernate.Session;
import org.hibernate.Transaction;
import org.springframework.orm.hibernate3.support.HibernateDaoSupport;

import com.wantall.framework.mvc.daosupport.hibernate.interfaces.IHibernateDAOSupport;

public class HibernateDAOSupport<T> extends HibernateDaoSupport implements IHibernateDAOSupport<T> {

    public void hibInsert(T obj) {
        getHibernateTemplate().save(obj);
        getHibernateTemplate().flush();
    }

    /**
     * 删除
     * 该方法必须要先查出T obj
     * @param id
     */
    public void hibDelete(Class<T> cl, Serializable id) {
        T obj = hibQuery(cl, id);
        if(obj != null) hibDelete(obj);
    }
    
    /**
     * 删除
     * 该方法可以删除一个new出来的对象 无需先查出
     * @param obj
     */
    public void hibDelete(T obj) {
        getHibernateTemplate().delete(obj);
        getHibernateTemplate().flush();
    }

    public void hibUpdate(T obj) {
        getHibernateTemplate().merge(obj);
        getHibernateTemplate().flush();
    } 
    
    public int excuteUpdateHQL(String hql, Object[] params) {
        int result = 0;
        Session session = null;
        Query query = null;
        Transaction tx = null;
        try {
            session = getSession();
            tx = session.beginTransaction();
            query = session.createQuery(hql);
            for(int i = 0 ; i < params.length ; i ++ ) {
                query.setParameter(i, params[i]);
            }
            result = query.executeUpdate();
            tx.commit();
        } catch(Exception e) {
            e.printStackTrace();
            tx.rollback();
        } finally {
            session.close();
            return result;
        }
    }
    
    public T hibQuery(Class<T> cl, Serializable id) {
        return (T) getHibernateTemplate().get(cl, id);
    }

    public List<T> hibQueryAllForList(Class<T> cl) {
        return this.getHibernateTemplate().loadAll(cl);
    }
    
    @SuppressWarnings("unchecked")
    public List<T> executeListQuery(String hql, Object... Params) {
        if (Params == null || Params.length == 0) {
            return this.getHibernateTemplate().find(hql);
        }
        return this.getHibernateTemplate().find(hql, Params);
    }
    
    /**
     * 简单hql查询,获取出的不一定是对象,可能是属性集合
     * @param hql
     * @return
     */
    @SuppressWarnings("unchecked")
    public List<T> hibCommonQuery(String hql) {
        Query query = this.getSession().createQuery(hql);
        return query.list();
    }

}

 

该类的实现是基于spring的hibernateDaoSupport的。

 

以上通过一个user的数据库访问操作,大致说明了j2eelib中数据库访问的一套基本机制,通过hibernate实现进行了简单介绍。本文强调的是设计观念, 因此在具体的实现类中,也会存在一些不完善的代码,或在IBaseDAO中缺乏一些笔者未曾想到的基本方法。大家可以指出哦(毕竟是技术交流嘛)。

 

    后续将继续介绍j2eelib中MVC设计中的其他相关设计。