通过这段时间的学习,深刻体会到entity,dao,service和controller是分层结构,从左到右依次为右边的层提供数据或服务。
第一个建立了entity,那么下边就该设计dao了。
泛型实现通用DAO的CURD操作
DAO对象的主要功能就是实现CURD操作,是实际开发中我们需要设计比较都的dao接口和实现类,但是里边有许多可复用的方法,避免重复定义。利用泛型语法,定义一个通用的dao对象解决代码复用的问题,下边描述使用Hibernate的ORM框架:
- 定义BaseDao 通用接口类
package com.shop.dao;
import java.io.Serializable;
import java.util.List;
import java.util.Map;
public interface BaseDao<T> {
public Serializable save(T o);
public void delete(Serializable id);
public void delete(T o);
public void update(T o);
public void saveOrUpdate(T o);
public T get(Serializable id);
public List<T> find(String hql);
public List<T> find(String hql, Map<String, Object> params );
public List<T> find(String hql, int page, int rows);
...
}
- dao接口和方法定义完了,接下来要实现BaseDAO<T>类了, 先创建实现类BaseDaoImpl 类:
import com.shop.dao.BaseDao;
public class BaseDaoImpl implements BaseDao<T>{
}
- 该类主要用于利用反射获取当前是哪个实体类,这样再通用对象的方法中我们就知道当前操作的是哪个实体类。因此我们定义一个clazz私有属性,并通过反射获取这个实体类对象。
//BaseDaoImpl类中添加如下代码
private Class<T> clazz;
@SuppressWarnings("unchecked")
public BaseDaoImpl() {
ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
this.clazz = (Class<T>) type.getActualTypeArguments()[0];
}
- BaseDaoImpl类中添加如下代码:
private SessionFactory sessionFactory;
public SessionFactory getSessionFactory() {
return sessionFactory;
}
@Autowired
public void setSessionFactory(SessionFactory sessionFactory) {
this.sessionFactory = sessionFactory;
}
- 完成上边几步,我们就可以实现接口中定义的方法了,对数据库的操作也可以使用sessionFactory属性来完成。
总结:我们的通用DAO的设计就大功告成了。下面来演示一下如何使用。我们只需要直接继承BaseDAO和BaseDaoImpl就可以拥有所有的方法了
@Entity//声明当前类为hibernate映射到数据库的实体类
@Table(name="User")//为实体Bean指定对应的数据表
public class User implements Serializable{
@Id //声明当前字段为主键
......
}
//创建用户DAO接口
public interface UserDao extends BaseDao<User> {
}
//实现UserDao接口
public class UserImpl extends BaseDaoImpl<User> implements UserDao{
}
dao中常用注解说明
@Repository("BaseDao")
注解是告诉Spring,让Spring创建一个名字叫“userDao”的UserDaoImpl实例。
当Service需要使用Spring创建的名字叫“userDao”的UserDaoImpl实例时,
就可以使用@Resource(name = "userDao")注解告诉Spring,Spring把创建好的userDao注入给Service即可。
@SuppressWarnings("all"):用于抑制编译器产生警告信息。
通过 @SuppressWarnings 的源码可知,其注解目标为类、字段、函数、函数入参、构造函数和函数的局部变量。
而作者建议注解应声明在最接近警告发生的位置。
常用类型:
抑制所有类型的警告:@SuppressWarnings("all")
抑制多类型的警告:@SuppressWarnings(value={"unchecked", "rawtypes"})
抑制单类型的警告:@SuppressWarnings("unchecked")