浅谈:java泛型与dao重用

时间:2023-03-08 17:58:03
浅谈:java泛型与dao重用

在进入今天的主题之前,我们先理解一下什么是泛型:

泛型是java中一种类型,泛型是被参数化的类型。

类型-->class

参数化-->class类型可以是任意参数

泛型存在的意义:泛型可以在编译的时候,告诉class传递的参数是什么类型,如果类型发送错误,在编译的时候,就会报错。

泛型所表现的形式:

1、java的list,set等集合所表现的。

    //这里面的E表示的就是泛型,
List<E> list = new ArrayList<E>();
//下面传入泛型的具体实例
List<Department> departmentList = new ArrayList<Department>();
//这个说明,在departmentList集合中,只能存放Department这一种类型

2、在类中的体现

package cn.ygh.boke;

public interface PersonDao<T> {
//这里的T也是泛型
}
package cn.ygh.boke;

public class PersonDaoImpl<T> implements PersonDao<T> {
//PersonDaoImpl这个类实现了PersonDao这个泛型类,那么他就要赋值给这个泛型类的泛型传值
/*
* PersonDaoImpl<T>的这个泛型T相当于给PersonDao<T>的T传值
* 总之,谁加泛型,谁给值
    这种方式也是我们下面主要介绍的方式
*/
}

3、

public class Person <T extends Collection<E>>{

}

介绍了什么三种泛型的表现形式,我们在介绍一下泛型的用法:泛型可以使用在某个类的属性或者方法(包括方法的参数和返回值)上

下面我来举一个比较实用的例子:例子基于hibernate,spring框架。

需求:要求写一个泛型的dao类,完成对所有对象进行简单的增删改查。

分析:我们在使用java操作数据库,因为hibernate的操作是基于对象,而我们从数据库中获取一行数据,也是需要把这行数据变成一个对象的

如果对每个不同的对象写一个xxxDao的类,把数据库的关系模型变成java的对象模型,实在是太过麻烦,能不能就写一个类,在类里面传入泛型,

让类根据泛型,泛型一个实例对象呢?

答案是可以的。

首先我们来写一个接口BaseDao,里面有增删改查的一些方法

//在这个类中,我们引入了泛型T,代表需要返回或者操作的java对象
public interface BaseDao<T> {
public Collection<T> getAllEntries();
public T getEntryById(Serializable id);
public void saveEntry(T t);
public void updateEntry(T t);
public void deleteEntryById(Serializable id);
}

然后我们再一个类BaseDaoImpl实现它,

import java.io.Serializable;
import java.lang.reflect.ParameterizedType;
import java.util.Collection; import org.springframework.orm.hibernate3.HibernateTemplate; import cn.itcast.oa.dao.base.BaseDao; //子类里面的T是给父类里面T赋值,子类里面的T是实参,父类里面的t是形参
public class BaseDaoImpl<T> implements BaseDao<T> { public HibernateTemplate hibernateTemplate;
private Class classt;//泛型的Class
/*
* 在父类中,要执行一段代码,执行的时间是在子类创建对象的时候,那么有两种解决方案
* 1、使用static代码块
* 2、利用父类的构造函数
*
* 分析:如果需要使用到this,那么就要使用父类的构造函数,如果不需要使用daothis,可以使用static代码块
* 因为下面需要使用this,获取ParameterizedType,所以使用父类的构造函数
* 如何获取泛型里面的class
*/
public BaseDaoImpl(){//使用父类的构造函数来实现对获取泛型的Class
//ParameterizedType,就是泛型,这里的this不是BaseDaoImpl,而是BaseDaoImpl的子类的this,
     //也就是一个类会继承BaseDaoImpl,然后通过它来获取其父类BaseDaoImpl的泛型T
ParameterizedType type = (ParameterizedType) this.getClass().getGenericSuperclass();
//获取到泛型的Class
this.classt = (Class) type.getActualTypeArguments()[0];
}   /*
  有人会问,为什么要获取泛型的Class,对应hibernate操作数据库来说,获取一个泛型的实例,需要通过泛型的Class来获取
  说白了,没有泛型的Class,就无法获取泛型的实例对象,也就没办法返回给service层实例对象
*/
public Collection<T> getAllEntries() {
return this.getHibernateTemplate().find("from "+this.classt.getName());
}

   //很明显的可以看出,我们的返回值使用的是T类型,不依赖任何的实例对象
public T getEntryById(Serializable id) {
// TODO Auto-generated method stub
return (T) this.hibernateTemplate.get(this.classt, id); }
   //这个方法也是,下面的方法都是这个特地,我就不异议介绍了
public void saveEntry(T t) {
// TODO Auto-generated method stub
this.hibernateTemplate.save(t);
} public void updateEntry(T t) {
// TODO Auto-generated method stub
this.hibernateTemplate.update(t);
} public void deleteEntryById(Serializable id) {
T t = this.getEntryById(id);
this.getHibernateTemplate().delete(t); } public HibernateTemplate getHibernateTemplate() {
return hibernateTemplate;
} public void setHibernateTemplate(HibernateTemplate hibernateTemplate) {
this.hibernateTemplate = hibernateTemplate;
}
}

下面来具体实现这个类,首先我们创建一个JavaBean:Department

import java.io.Serializable;
import java.util.Set; public class Department implements Serializable{ private Long did;//部门id
private String dname;//部门名称
private String description;//部门描述
private Set<User> users;//部门里面所有的用户 public Long getDid() {
return did;
}
public void setDid(Long did) {
this.did = did;
}
public String getDname() {
return dname;
}
public void setDname(String dname) {
this.dname = dname;
}
public String getDescription() {
return description;
}
public void setDescription(String description) {
this.description = description;
}
public Set<User> getUsers() {
return users;
}
public void setUsers(Set<User> users) {
this.users = users;
} }

创建一个接口DepartmentDao<T>接口,里面有对Department操作的方法,让它继承BaseDao<T>这个接口

import java.io.Serializable;
import java.util.Collection; import cn.itcast.oa.dao.base.BaseDao;
import cn.itcast.oa.domain.Department; public interface DepartmentDao<T> extends BaseDao<T>{
//对department操作的各种方法
public void saveDepartment(Department department);
public void updateDepartment(Department department);
public void deleteDepartmentById(Serializable id,String deleteMode);
public Collection<Department> getAllDepartments();
public Department getDepartmentById(Serializable id);
}

在创建DepartmentDao的一个子类DepartmentDaoImpl,实现DepartmentDao,继承BaseDaoImpl,把泛型传递过去

这样这个DepartmentDaoImpl,就可以操作对department的增删改查(只是简单的实现)。

到这里,我们可以发现,BaseDaoImpl<Department> 和DepartmentDao<Department>可以无限次重用,只要往这两个类传递泛型

那么就可以直接完成对泛型对象的增删改查的操作。

import java.io.Serializable;
import java.util.Collection;
import java.util.Set; import org.springframework.orm.hibernate3.support.HibernateDaoSupport; import cn.itcast.oa.dao.DepartmentDao;
import cn.itcast.oa.dao.base.impl.BaseDaoImpl;
import cn.itcast.oa.domain.Department;
import cn.itcast.oa.domain.User;
import cn.itcast.oa.utils.DeleteModel; //传递泛型
public class DepartmentDaoImpl extends BaseDaoImpl<Department> implements DepartmentDao<Department>{ public void saveDepartment(Department department) {
//直接使用父类的泛型方法进行操作
// TODO Auto-generated method stub
this.saveEntry(department);
} public void updateDepartment(Department department) {
//直接使用父类的泛型方法进行操作
// TODO Auto-generated method stub
this.updateEntry(department);
} public void deleteDepartmentById(Serializable id,String deleteMode) {
// TODO Auto-generated method stub
Department department = (Department) this.getHibernateTemplate().get(Department.class, id);
if(DeleteModel.DEL.equals(deleteMode)){
this.getHibernateTemplate().delete(department);
}else if(DeleteModel.DEL_PRE_RELEASE.equals(deleteMode)){
Set<User> users = department.getUsers();
for(User user:users){
user.setDepartment(null);//解除关系
}
this.getHibernateTemplate().flush();
this.getHibernateTemplate().delete(department);
}else{//进行级联删除
Set<User> users = department.getUsers();
for(User user:users){
user.setDepartment(null);//解除关系部门关系
user.setPosts(null);//解除岗位关系
this.getHibernateTemplate().flush();
this.getHibernateTemplate().delete(user);
}
//删除部门
this.getHibernateTemplate().delete(department);
} } public Collection<Department> getAllDepartments() {
//直接使用父类的泛型方法进行操作
return this.getAllEntries();
} public Department getDepartmentById(Serializable id) {
//直接使用父类的泛型方法进行操作
return this.getEntryById(id);
} }

泛型的知识到这里就已经结束了,下面赋上hibernate配置文件和spring配置文件,让大家可以更好的理解

Department.hbm.xml:Department的映射文件

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd"> <hibernate-mapping>
<class name="cn.itcast.oa.domain.Department" table="department">
<id name="did" type="java.lang.Long" length="5">
<generator class="increment"></generator>
</id>
<property name="dname" type="java.lang.String" length="20"></property>
<property name="description" type="java.lang.String" length="500"></property>
<set name="users" inverse="true">
<key>
<column name="did"></column>
</key>
<one-to-many class="cn.itcast.oa.domain.User"/>
</set>
</class>
</hibernate-mapping>

applicationContext.xml:spring的总的配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
<!-- 导入其他的spring配置文件-->
<import resource="applicationContext-db.xml"/>
<import resource="applicationContext-person.xml"/>
<import resource="applicationContext-department.xml"/>
</beans>

applicationContext-db.xml:spring的关于hibernate和数据库的配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-2.5.xsd"> <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
<property name="configLocation">
<value>classpath:hibernate/hibernate.cfg.xml</value>
</property>
</bean>
<bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager">
<property name="sessionFactory">
<ref bean="sessionFactory"/>
</property>
</bean> <bean id="hibernateTemplate" class="org.springframework.orm.hibernate3.HibernateTemplate">
<property name="sessionFactory">
<ref bean="sessionFactory"/>
</property>
</bean> <tx:advice transaction-manager="transactionManager" id="tx">
<tx:attributes>
<tx:method name="save*" read-only="false"/>
<tx:method name="update*" read-only="false"/>
<tx:method name="delete*" read-only="false"/>
<tx:method name="*" read-only="true"/>
</tx:attributes>
</tx:advice> <aop:config>
<aop:pointcut expression="execution(* cn.itcast.oa.service.impl.*.*(..))" id="perform"/>
<aop:advisor advice-ref="tx" pointcut-ref="perform"/>
</aop:config> <!-- <bean id="baseDdaoImpl" class="cn.itcast.oa.dao.base.impl.BaseDaoImpl" abstract="true">
<property name="hibernateTemplate">
<ref bean="hibernateTemplate"/>
</property>
</bean> -->
</beans>

applicationContext-department.xml:spring的关于department的配置文件。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-2.5.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
<bean id="departmentDao" class="cn.itcast.oa.dao.impl.DepartmentDaoImpl">
<property name="hibernateTemplate">
<ref bean="hibernateTemplate"/>
</property>
</bean> <bean id="departmentService" class="cn.itcast.oa.service.impl.DepartmentServiceImpl">
<property name="departmentDao">
<ref bean="departmentDao"/>
</property>
</bean>
<bean id="departmentAction" class="cn.itcast.oa.struts.action.DepartmentAction" scope="prototype">
<property name="departmentService">
<ref bean="departmentService"/>
</property>
</bean> </beans>

希望对大家有帮助。