泛型是Java SE 1.5的新特性, 泛型的本質是參數化類型, 也就是說所操作的數據類型被指定為一個參數. 因此我們可以利用泛型和反射來設計一些通用方法. 現在有2張表, 一張user表和一張student表.
user:

student:

如果要根據id查詢數據, 你會怎麼做呢?寫2個方法分別查詢user和student?其實這時候我們就可以使用泛型和反射寫一個通用的方法.
user的實體類:

private Integer id;
private String username;
private String password;
private String hobby;
//getXxx方法和setXxx方法
View Code
student實體類:

private Integer id;
private String name;
private Integer age;
//getXxx方法和setXxx方法
View Code
BaseDao接口:

public interface BaseDao<T> {
//根據id查詢的方法
T findById(Integer id);
}
View Code
BaseDaoImpl類, 實現了BaseDao接口, 通用方法就在這裡面完成:
//設置為抽象的, 不讓他實例化, 讓其子類實例化就行了
//通過泛型設計通用方法的關鍵就是利用反射拿到泛型的具體類型
public abstract class BaseDaoImpl<T> implements BaseDao<T> {
private String tableName; //表名
private Class<T> actualType;//真實類型
/**
* findById(Integer id)這個方法被子類繼承了, 假設我們在UserDaoImpl中操作, 此時參數化類型T為User
* 下面的講解都假設是在UserDaoImpl中進行的
*/
//把公共部分可以放到構造方法中
@SuppressWarnings("unchecked")
public BaseDaoImpl() {
//返回類型是Type 是 Java 編程語言中所有類型的公共高級接口. 它們包括原始類型、參數化類型、數組類型、類型變量和基本類型.
//Type的已知子接口: ParameterizedType 表示參數化類型, 如 Collection<String>.
//getClass()得到UserDaoImpl的Class, 得到Class一般有3中方式: getClass(), 類名.class, Class.forName()
Type type = getClass().getGenericSuperclass();//獲取UserDaoImpl<User>的參數化類型的父類BaseDaoImpl<User>
//由於我們得到的是一個參數化類型, 所以轉成ParameterizedType, 因為需要使用裡面的方法
ParameterizedType pt = (ParameterizedType) type;//強轉
Type[] actualTypeArr = pt.getActualTypeArguments();//獲取真實參數類型數組[User.class], 可以調試看到這裡的值
actualType = (Class<T>) actualTypeArr[0];//數組只有一個元素
tableName = actualType.getSimpleName();
}
@Override
public T findById(Integer id) {
String sql = "select * from " + tableName + " where id = ?";
try {
return QRUtils.getQueryRunner().query(sql, new BeanHandler<T>(actualType), id);
} catch (SQLException e) {
e.printStackTrace();
}
return null;
}
}
連接數據庫操作是用的c3p0和dbutils, 有關這方面的內容可以參看我以前的隨筆.
UserDao接口, 繼承BaseDao接口:

public interface UserDao<T> extends BaseDao<T> {
}
View Code
UserDaoImpl類, 實現UserDao接口, 繼承BaseDaoImpl類, 可以看到裡面什麼方法也沒有:

public class UserDaoImpl extends BaseDaoImpl<User> implements UserDao<User> {
}
View Code
StudentDao接口, 繼承BaseDao接口:

public interface StudentDao<T> extends BaseDao<T> {
}
View Code
StudentDaoImpl類, 實現StudentDao接口, 繼承BaseDaoImpl類, 也可以看到裡面什麼方法也沒有:

public class StudentDaoImpl extends BaseDaoImpl<Student> implements StudentDao<Student> {
}
View Code
從以上dao可以看到, 我是在繼承BaseDaoImpl類時把泛型具體化了.
測試:
@Test
public void testGeneric() throws Exception {
UserDao<User> userDao = new UserDaoImpl();
System.out.println(userDao.findById(1));
System.out.println("-------------------");
StudentDao<Student> studentDao = new StudentDaoImpl();
System.out.println(studentDao.findById(1));
}
看一下測試的結果: 同一個方法把2張表中的數據都查出來了

有關泛型的基本使用, 請參考java泛型基礎