java泛型基本知識(shí)及通用方法
泛型的基本使用
泛型是Java SE 1.5的新特性, 泛型的本質(zhì)是參數(shù)化類(lèi)型, 也就是說(shuō)所操作的數(shù)據(jù)類(lèi)型被指定為一個(gè)參數(shù). 這種參數(shù)類(lèi)型可以用在類(lèi)、接口和方法的創(chuàng)建中, 分別稱(chēng)為泛型類(lèi)、泛型接口、泛型方法. Java語(yǔ)言引入泛型的好處是安全簡(jiǎn)單.
今天就從以下幾個(gè)方面介紹一下java的泛型: 基礎(chǔ), 泛型關(guān)鍵字, 泛型方法, 泛型類(lèi)和接口.
基礎(chǔ):
通過(guò)集合的泛型了解泛型的基本使用
public void testBasis(){
List<String> list = new ArrayList<String>();
// new ArrayList<int>();
}
//這是最基本的泛型使用, 就不多說(shuō)了, 不過(guò)要注意的是泛型只能是引用數(shù)據(jù)類(lèi)型, 不能是基本類(lèi)型, 而且泛型只在編譯期有效, 在編譯后的class文件中是不存在泛型信息的
注意: 泛型只在編譯期有效, 在編譯后的class文件中是不存在泛型信息的
泛型關(guān)鍵字:
通配符?表示任意引用類(lèi)型, extends關(guān)鍵字表示子類(lèi)及其本身, super關(guān)鍵字是表示父類(lèi)及其本身. 通過(guò)一個(gè)例子看一下, 解釋說(shuō)明都在例子中
public void testKeyWord() throws Exception {
//實(shí)例化參數(shù)類(lèi)型必須指明具體類(lèi)型
List<?> list = new ArrayList<String>();
//由于list中類(lèi)型不明確, 所以不能進(jìn)行添加操作
// list.add("sk");
//通配符?表示任意引用類(lèi)型
List<List<?>> list1 = new ArrayList<List<?>>();
//list1的參數(shù)化類(lèi)型是一個(gè)List, 而這個(gè)List也是一個(gè)參數(shù)化類(lèi)型, 它的類(lèi)型是任意類(lèi)型, 所以list1可以添加任意List類(lèi)型對(duì)象
list1.add(new ArrayList<Object>());
list1.add(new ArrayList<String>());
//實(shí)例化list2時(shí)指明了類(lèi)型參數(shù)List, 只不過(guò)這個(gè)List是一個(gè)泛型類(lèi)型
//從這里可以看到關(guān)鍵字extends的用法, 其實(shí)就是繼承, 如下這里的list2中的參數(shù)化類(lèi)型List(在這里記為paramList)的參數(shù)化類(lèi)型是繼承自Number的
//所以在list2在添加的時(shí)候只能添加參數(shù)化類(lèi)型為Number及其子類(lèi)的paramList
List<Integer> l1 = new ArrayList<Integer>();
List<Number> l2 = new ArrayList<Number>();
List<Object> l3 = new ArrayList<Object>();
List<List<? extends Number>> list2 = new ArrayList<List<? extends Number>>();
list2.add(l1);
list2.add(l2);
// list2.add(l3); //這里使用了extends關(guān)鍵字, 所以不能添加Number的父類(lèi)
//相信大家也猜到了, 既然extends關(guān)鍵字表示子類(lèi)及其本身, 那么super關(guān)鍵字是表示父類(lèi)及其本身, 是的, 沒(méi)錯(cuò)
//和上面的不一樣了, l1不能添加, 因?yàn)镮nteger是Number的子類(lèi), 并不是Number的父類(lèi)
List<List<? super Number>> list3 = new ArrayList<List<? super Number>>();
// list3.add(l1); //這里使用了super關(guān)鍵字, 所以不能添加Number的子類(lèi)
list3.add(l2);
list3.add(l3);
}
泛型方法:
java中任何類(lèi)型必須先定義才能使用, 泛型也是如此. 既然要使用泛型作為參數(shù), 所以要先定義, 泛型的定義在訪問(wèn)修飾符和返回類(lèi)型之間, 注意不要掉了尖括號(hào)
//無(wú)返回值有參的方法, 參數(shù)為泛型
public <T> void show(T t){
System.out.println("t-=-=" + t);
}
//有返回值的有參方法, 只有一個(gè)參數(shù)化類(lèi)型, 這里定義泛型的方式和上面一樣, 也是先定義后使用, 只不過(guò)這里的返回類(lèi)型是泛型
public <T> T get(T t){
return t;
}
//有返回值的有參方法, 有多個(gè)參數(shù)化類(lèi)型, 這里以兩個(gè)為例
public <T, K> K get(T t, K k){
return k;
}
@Test
public void testGeneric() throws Exception {
show(3);
show("generic");
System.out.println("----------------");
System.out.println("我返回Integer類(lèi)型-" + get(4));
System.out.println("我返回String類(lèi)型-" + get("returnGeneric"));
System.out.println("------------------");
System.out.println("我返回String類(lèi)型-" + get(1, "a"));
System.out.println("我返回Integer類(lèi)型-" + get("b", 2));
}
本來(lái)想寫(xiě)無(wú)參的泛型方法, 可是寫(xiě)著寫(xiě)著感覺(jué)那樣沒(méi)有什么意義, 不知道各位有什么見(jiàn)解.
泛型類(lèi)和接口:
寫(xiě)泛型類(lèi)的時(shí)候只需要在類(lèi)名后面加上泛型即可, 就像這樣
public class GenericClass<T> {
public T get(T t){
return t;
}
public void scr(T t){
System.out.println(t);
}
public void show(){
GenericClass<Integer> gc = new GenericClass<Integer>();
// GenericClass<T> gc1 = new GenericClass<T>();
gc.get(3);
gc.scr(5);
//下面2個(gè)會(huì)報(bào)錯(cuò)
// gc1.get(3);
// gc1.scr(5);
}
}
從上面的例子中可以看到, 參數(shù)化類(lèi)型是在創(chuàng)建對(duì)象的時(shí)候具體化的, 那么除此之外, 還可以再什么時(shí)候具體化參數(shù)化類(lèi)型呢?
如果是在繼承或?qū)崿F(xiàn)中, 可以在子類(lèi)或?qū)崿F(xiàn)類(lèi)中確定具體類(lèi)型
使用java泛型設(shè)計(jì)通用方法
泛型是Java SE 1.5的新特性, 泛型的本質(zhì)是參數(shù)化類(lèi)型, 也就是說(shuō)所操作的數(shù)據(jù)類(lèi)型被指定為一個(gè)參數(shù). 因此我們可以利用泛型和反射來(lái)設(shè)計(jì)一些通用方法. 現(xiàn)在有2張表, 一張user表和一張student表.
user:

student:

如果要根據(jù)id查詢數(shù)據(jù), 你會(huì)怎么做呢?寫(xiě)2個(gè)方法分別查詢user和student?其實(shí)這時(shí)候我們就可以使用泛型和反射寫(xiě)一個(gè)通用的方法.
user的實(shí)體類(lèi):
private Integer id; private String username; private String password; private String hobby; //getXxx方法和setXxx方法
student實(shí)體類(lèi):
private Integer id; private String name; private Integer age; //getXxx方法和setXxx方法
BaseDao接口:
public interface BaseDao<T> {
//根據(jù)id查詢的方法
T findById(Integer id);
}
BaseDaoImpl類(lèi), 實(shí)現(xiàn)了BaseDao接口, 通用方法就在這里面完成:
//設(shè)置為抽象的, 不讓他實(shí)例化, 讓其子類(lèi)實(shí)例化就行了
//通過(guò)泛型設(shè)計(jì)通用方法的關(guān)鍵就是利用反射拿到泛型的具體類(lèi)型
public abstract class BaseDaoImpl<T> implements BaseDao<T> {
private String tableName; //表名
private Class<T> actualType;//真實(shí)類(lèi)型
/**
* findById(Integer id)這個(gè)方法被子類(lèi)繼承了, 假設(shè)我們?cè)赨serDaoImpl中操作, 此時(shí)參數(shù)化類(lèi)型T為User
* 下面的講解都假設(shè)是在UserDaoImpl中進(jìn)行的
*/
//把公共部分可以放到構(gòu)造方法中
@SuppressWarnings("unchecked")
public BaseDaoImpl() {
//返回類(lèi)型是Type 是 Java 編程語(yǔ)言中所有類(lèi)型的公共高級(jí)接口. 它們包括原始類(lèi)型、參數(shù)化類(lèi)型、數(shù)組類(lèi)型、類(lèi)型變量和基本類(lèi)型.
//Type的已知子接口: ParameterizedType 表示參數(shù)化類(lèi)型, 如 Collection<String>.
//getClass()得到UserDaoImpl的Class, 得到Class一般有3中方式: getClass(), 類(lèi)名.class, Class.forName()
Type type = getClass().getGenericSuperclass();//獲取UserDaoImpl<User>的參數(shù)化類(lèi)型的父類(lèi)BaseDaoImpl<User>
//由于我們得到的是一個(gè)參數(shù)化類(lèi)型, 所以轉(zhuǎn)成ParameterizedType, 因?yàn)樾枰褂美锩娴姆椒?
ParameterizedType pt = (ParameterizedType) type;//強(qiáng)轉(zhuǎn)
Type[] actualTypeArr = pt.getActualTypeArguments();//獲取真實(shí)參數(shù)類(lèi)型數(shù)組[User.class], 可以調(diào)試看到這里的值
actualType = (Class<T>) actualTypeArr[0];//數(shù)組只有一個(gè)元素
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;
}
}
連接數(shù)據(jù)庫(kù)操作是用的c3p0和dbutils, 有關(guān)這方面的內(nèi)容可以參看我以前的隨筆.
UserDao接口, 繼承BaseDao接口:
public interface UserDao<T> extends BaseDao<T> {
}
UserDaoImpl類(lèi), 實(shí)現(xiàn)UserDao接口, 繼承BaseDaoImpl類(lèi), 可以看到里面什么方法也沒(méi)有:
public class UserDaoImpl extends BaseDaoImpl<User> implements UserDao<User> {
}
StudentDao接口, 繼承BaseDao接口:
public interface StudentDao<T> extends BaseDao<T> {
}
StudentDaoImpl類(lèi), 實(shí)現(xiàn)StudentDao接口, 繼承BaseDaoImpl類(lèi), 也可以看到里面什么方法也沒(méi)有:
public class StudentDaoImpl extends BaseDaoImpl<Student> implements StudentDao<Student> {
}
從以上dao可以看到, 我是在繼承BaseDaoImpl類(lèi)時(shí)把泛型具體化了.
測(cè)試:
@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));
}
看一下測(cè)試的結(jié)果: 同一個(gè)方法把2張表中的數(shù)據(jù)都查出來(lái)了

相關(guān)文章
Spring Cloud重試機(jī)制與各組件的重試總結(jié)
這篇文章主要給大家介紹了關(guān)于Spring Cloud中重試機(jī)制與各組件的重試的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11
使用Spring的攔截器監(jiān)測(cè)每個(gè)Controller或方法的執(zhí)行時(shí)長(zhǎng)
這篇文章主要介紹了使用Spring的攔截器監(jiān)測(cè)每個(gè)Controller或方法的執(zhí)行時(shí)長(zhǎng),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10
Liquibase結(jié)合SpringBoot使用實(shí)現(xiàn)數(shù)據(jù)庫(kù)管理功能
Liquibase 是一個(gè)強(qiáng)大的數(shù)據(jù)庫(kù)管理工具,它幫助你通過(guò)自動(dòng)化管理數(shù)據(jù)庫(kù)的變更、版本控制、和回滾,簡(jiǎn)化了開(kāi)發(fā)中的數(shù)據(jù)庫(kù)遷移工作,這篇文章主要介紹了Liquibase結(jié)合SpringBoot使用實(shí)現(xiàn)數(shù)據(jù)庫(kù)管理,需要的朋友可以參考下2024-12-12
關(guān)于MyBatis通用Mapper@Table注解使用的注意點(diǎn)
這篇文章主要介紹了關(guān)于MyBatis通用Mapper@Table注解使用的注意點(diǎn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
基于SpringBoot+vue實(shí)現(xiàn)前后端數(shù)據(jù)加解密
這篇文章主要給大家介紹了基于SpringBoot+vue實(shí)現(xiàn)前后端數(shù)據(jù)加解密,文中有詳細(xì)的示例代碼,具有一定的參考價(jià)值,感興趣的小伙伴可以自己動(dòng)手試一試2023-08-08
Java數(shù)據(jù)結(jié)構(gòu)與算法學(xué)習(xí)之雙向鏈表
雙向鏈表也叫雙鏈表,是鏈表的一種,它的每個(gè)數(shù)據(jù)結(jié)點(diǎn)中都有兩個(gè)指針,分別指向直接后繼和直接前驅(qū)。所以,從雙向鏈表中的任意一個(gè)結(jié)點(diǎn)開(kāi)始,都可以很方便地訪問(wèn)它的前驅(qū)結(jié)點(diǎn)和后繼結(jié)點(diǎn)。本文將為大家詳細(xì)介紹雙向鏈表的特點(diǎn)與使用,需要的可以參考一下2021-12-12
Spring Security+Spring Data Jpa如何進(jìn)行安全管理
這篇文章主要介紹了Spring Security+Spring Data Jpa如何進(jìn)行安全管理,幫助大家更好的理解和學(xué)習(xí)Spring Security框架,感興趣的朋友可以了解下2020-09-09

