Java:反射+泛型:獲取類型參數的實例

2019獨角獸企業重金招聘Python工程師標準>>> hot3.png

場景描述:

我需要開發四個頁面,每個頁面都是只涉及增刪改查的基本邏輯。

最簡單的寫法:

創建四個接口A,B,C,D,每個接口中都聲明了增刪改查四個方法,完全一致

public Map<String,Object> delete(HttpServletRequest request, User user);public Map<String,Object> query(HttpServletRequest request, User user, Map<String, Object> params);public Map<String,Object> insert(HttpServletRequest request, User user);public Map<String,Object> update(HttpServletRequest request, User user);

為上面四個接口分別創建四個實現類,復寫接口中的增刪改查方法

接口A的實現類

@Override
public Map<String, Object> delete(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "delete");return commandExecutor.execute(new DaoAImpl(request, params, user));
}
@Override
public Map<String, Object> query(HttpServletRequest request, User user,	Map<String, Object> params) {params = null == params ? new HashMap<String,Object>() : params;params.put("operation", "query");return commandExecutor.execute(new DaoAImpl(request, params, user));
}
@Override
public Map<String, Object> insert(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "insert");return commandExecutor.execute(new DaoAImpl(request, params, user));
}
@Override
public Map<String, Object> update(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "update");return commandExecutor.execute(new DaoAImpl(request, params, user));
}

接口B的實現類

@Override
public Map<String, Object> delete(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "delete");return commandExecutor.execute(new DaoBImpl(request, params, user));
}
@Override
public Map<String, Object> query(HttpServletRequest request, User user,	Map<String, Object> params) {params = null == params ? new HashMap<String,Object>() : params;params.put("operation", "query");return commandExecutor.execute(new DaoBImpl(request, params, user));
}
@Override
public Map<String, Object> insert(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "insert");return commandExecutor.execute(new DaoBImpl(request, params, user));
}
@Override
public Map<String, Object> update(HttpServletRequest request, User user) {Map<String,Object> params = new HashMap<String,Object>();params.put("operation", "update");return commandExecutor.execute(new DaoBImpl(request, params, user));
}

接口C,D 的實現類就不在展示了。其中主要區別就是:

ServiceAImpl調用DaoAImpl的對象

ServiceBImpl調用DaoBImpl的對象

ServiceCImpl調用DaoCImpl的對象

ServiceDImpl調用DaoDImpl的對象

看完上面的代碼,你可以發現,重復的代碼太多了,需要整理下

解決方案:

接口:可以將增刪改查的方法聲明放到一個基礎接口中,然后繼承這個基礎接口就行。

基礎service接口

package com.engine.odoc.service;import java.util.Map;import javax.servlet.http.HttpServletRequest;import weaver.hrm.User;public interface BaseService {public Map<String,Object> delete(HttpServletRequest request, User user);public Map<String,Object> query(HttpServletRequest request, User user, Map<String, Object> params);public Map<String,Object> insert(HttpServletRequest request, User user);public Map<String,Object> update(HttpServletRequest request, User user);
}

繼承基礎service接口的接口

package com.engine.odoc.service;public interface ServiceA extends BaseService {}
package com.engine.odoc.service;public interface ServiceB extends BaseService {}
package com.engine.odoc.service;public interface ServiceC extends BaseService {}
package com.engine.odoc.service;public interface ServiceD extends BaseService {}

接口的實現:同樣是增刪改查四個復寫的方法,唯一不同的就是不同的實現類調用不同的持久層(DAO)對象。期初的想法與處理接口的想法一樣,把復寫的增刪改查方法寫到一個基礎service實現類中,然后繼承這個基礎service實現類即可。

為了能實現這個功能,我們需要用到Java中的泛型和反射知識

新建一個基礎service實現類

package com.engine.odoc.service.impl;public class BaseServiceImpl<T> implements BaseService {@Overridepublic Map<String, Object> delete(HttpServletRequest request, User user) {}@Overridepublic Map<String, Object> query(HttpServletRequest request, User user,	Map<String, Object> params) {}@Overridepublic Map<String, Object> insert(HttpServletRequest request, User user) {}@Overridepublic Map<String, Object> update(HttpServletRequest request, User user) {}}

可以看到這個基礎service實現類接受了一個類型參數T

那些繼承了這個基礎實現類的類,就可以通過這個類型參數,把需要調用的Dao層對象的類型傳遞到父類當中
?

package com.engine.odoc.service.impl;public class ServiceAImpl extends BaseServiceImpl<DaoAImpl> implements ServiceA {}
package com.engine.odoc.service.impl;public class ServiceBImpl extends BaseServiceImpl<DaoBImpl> implements ServiceB {}
package com.engine.odoc.service.impl;public class ServiceCImpl extends BaseServiceImpl<DaoCImpl> implements ServiceC {}
package com.engine.odoc.service.impl;public class ServiceDImpl extends BaseServiceImpl<DaoDImpl> implements ServiceD {}

?

接下來,我們需要思考的是,如何通過這個類型參數T,來得到T的一個實例。

我們先上菜

package com.engine.odoc.service.impl;import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;
import java.util.HashMap;
import java.util.Map;import javax.servlet.http.HttpServletRequest;import com.engine.odoc.service.BaseService;public class BaseServiceImpl<T> implements BaseService {public Class cusClass;public BaseServiceImpl() {// 獲取T.classType genericSuperclass = this.getClass().getGenericSuperclass();if(genericSuperclass instanceof ParameterizedType) {ParameterizedType parameterizedType = (ParameterizedType) genericSuperclass;Type[] typeArray = parameterizedType.getActualTypeArguments();if(null != typeArray && typeArray.length>0) {cusClass = (Class) typeArray[0];}}}/*** 獲取T類的實例* @param request* @param params* @param user* @return*/private T getTInstance(HttpServletRequest request, Map<String,Object> params, User user) {try {return (T) cusClass.getConstructor(new Class[]{HttpServletRequest.class, Map.class, User.class}).newInstance(request, params, user);} catch (Exception e) {e.printStackTrace();}return null;}@Overridepublic Map<String, Object> delete(HttpServletRequest request, User user) {}@Overridepublic Map<String, Object> query(HttpServletRequest request, User user,	Map<String, Object> params) {}@Overridepublic Map<String, Object> insert(HttpServletRequest request, User user) {}@Overridepublic Map<String, Object> update(HttpServletRequest request, User user) {}}

?

????????我們可以看到,我們在這個基礎service實現類的默認構造函數中確認了類型參數T的類型,得到T.class。之后在getTInstance() 方法中通過反射的方式獲取了T的實例。

?

Type genericSuperclass = this.getClass().getGenericSuperclass();

????????這行代碼我們是寫在父類(BaseServiceImpl)的默認構造函數中的。繼承這個類的子類們,在執行默認構造函數的時候,會先執行super();調用父類的默認構造函數,這時候,以上這行代碼中的this就代表了子類,調用getGenericSuperclass()來獲得父類(BaseServiceImpl),而得到的父類(BaseServiceImpl)可能是BaseServiceImpl<DaoAImpl>,BaseServiceImpl<DaoBImpl>... ,也可能BaseServiceImpl<T>。

我們可以使用接口ParameterizedType,用來檢驗類型參數是否被參數化

Type genericSuperclass = this.getClass().getGenericSuperclass();
if(genericSuperclass instanceof ParameterizedType) {// 該泛型類的類型參數已經被參數化
}

如果類型參數已經被參數化,我們就可以通過調用下面的方法

package java.lang.reflect;public interface ParameterizedType extends Type {Type[] getActualTypeArguments();
}

返回的數組中就存儲了類型參數T的具體類型,即T.class

?

獲取到T.class之后,我們就可以通過反射來進一步獲得T的實例

try {return (T) cusClass.getConstructor(new Class[]{HttpServletRequest.class, Map.class, User.class}).newInstance(request, params, user);
} catch (Exception e) {e.printStackTrace();
}

?

?

?

?

轉載于:https://my.oschina.net/u/3229807/blog/1821207

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/277909.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/277909.shtml
英文地址,請注明出處:http://en.pswp.cn/news/277909.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

servlet(1)

servlet類分級&#xff1a; 1.ServletConfig接口類&#xff1a;理解為讀取servlet配置的類&#xff0c;里面有四個抽象方法如下&#xff1a; ①getServletName:獲取servlet在web.xml中的名字 ②getServletContext&#xff1a;獲取Servlet上下文&#xff0c;相當于web項目管理所…

如何在 Apple Silicon (M1) 上開發 Teams App

apple 在幾個月前發布了自家的芯片 M1&#xff0c;由于將多核cpu&#xff0c;多核gpu&#xff0c;神經網絡運算&#xff0c;內存和其他一切處理部件高度整合在一起&#xff0c;大大提高數據傳輸速度。發布后好評如潮&#xff0c;我也沒有忍住&#xff0c;入手了一臺最低配的mac…

BZOJ5289 洛谷4437:[HNOI/AHOI2018]排列——題解

https://www.lydsy.com/JudgeOnline/problem.php?id5289 https://www.luogu.org/problemnew/show/P4437 考慮對于a[i]m,a[m]n&#xff0c;我們令p[j]i,p[k]m&#xff08;一定會有一對(j,k)滿足這個條件的&#xff09;&#xff0c;則我們會有p[k]a[p[j]]&#xff0c;此時我們要…

集成學習-Adaboost

Adaboost 中文名叫自適應提升算法&#xff0c;是一種boosting算法。 boosting算法的基本思想 對于一個復雜任務來說&#xff0c;單個專家的決策過于片面&#xff0c;需要集合多個專家的決策得到最終的決策&#xff0c;通俗講就是三個臭皮匠頂個諸葛亮。 對于給定的數據集&#…

主動給團隊或用戶安裝Teams App

在寫這篇文章的時候&#xff0c;這個新功能還處在 Public Review&#xff0c;這意味著可能&#xff08;很小的可能性&#xff09;這里寫的方法在正式發布前還會有一些改動。 之前有一些做teams app開發的朋友問過我&#xff0c;能不能主動給一個team或者一個用戶安裝一個指定的…

thinkphp5多級控制器是什么?怎么使用?

thinkphp5多級控制器是什么&#xff1f;怎么使用&#xff1f; 一、總結 1、多級控制器是讓控制器的級數變成多級&#xff0c;也就是controller目錄下可以新建其它目錄。 2、使用的話注意目錄下的控制的的命名空間&#xff08;加上目錄名&#xff09;&#xff08;namespace app\…

給Teams消息附加圖片的三種方式

Teams消息支持三種不同的方式來添加圖片&#xff0c;這篇文章我們來一起看一下這三種方式。 Inline圖片 var imagePath Path.Combine(Environment.CurrentDirectory, "abc.png"); var imageData Convert.ToBase64String(File.ReadAllBytes(imagePath)); var image…

4月18日 MySQL學習

正式開始了數據庫的學習 昨天下好的MySQL 今天正式開始學習的&#xff0c;介紹了多種數據庫軟件&#xff0c;當然 學習的這個是開源的 免費的。 DBMS(數據庫管理系統)這就是我們學習的數據庫的軟件 數據庫分為關系型數據庫管理系統和非關系型數據庫管理系統(沒有深入的了解) 今…

企業數據湖構建之旅

摘要&#xff1a;隨著互聯網的發展&#xff0c;數據的規模和類型都呈現一個爆炸性的增長&#xff0c;對于這么多類型的數據&#xff0c;如何進行有效的管理和存儲&#xff0c;包括數據的分析&#xff0c;這是大家要面臨的一個問題。在武漢云棲大會上&#xff0c;阿里云高級產品…

用AzureFunction開發最簡單的Teams Bot

之前我有一篇文章講了如何在azure function上開發最簡單的outgoing webhook&#xff0c;收到一些反饋&#xff0c;建議我介紹一下如果在azure function上開發teams bot&#xff0c;那這篇文章就來講一下如何用function來快速開發bot。 我們先創建一個azure function資源&#…

20189215 2018-2019-2 《密碼與安全新技術專題》第7周作業

課程&#xff1a;《密碼與安全新技術專題》 班級&#xff1a; 1892班 姓名&#xff1a; 李煬 學號&#xff1a;20189215 上課教師&#xff1a;謝四江 上課日期&#xff1a;2019年4月9日 必修/選修&#xff1a; 選修 1.本次講座的學習總結 講座主題&#xff1a;信息隱藏 信息隱藏…

BZOJ1565[NOI2009]植物大戰僵尸——最大權閉合子圖+拓撲排序

題目描述 Plants vs. Zombies&#xff08;PVZ&#xff09;是最近十分風靡的一款小游戲。Plants&#xff08;植物&#xff09;和Zombies&#xff08;僵尸&#xff09;是游戲的主角&#xff0c;其中Plants防守&#xff0c;而Zombies進攻。該款游戲包含多種不同的挑戰系列&#xf…

推送ActivityFeed到Teams

幾個月前&#xff0c;Teams 團隊又推出了新的 Graph API&#xff0c;讓 app 可以給用戶發送 Activity Feed。我們來看看如何做。 首先&#xff0c;我們的app需要使用較新的 manifest 1.7版本&#xff0c;當然如果使用最新的1.8版本就更好了。在manifest json中添加 webApplica…

RecycleView彈性滑動

還有點bug&#xff0c;建議使用 LinearSnapHelper rvPilotList.addOnScrollListener(new RecyclerView.OnScrollListener() {Overridepublic void onScrolled(NonNull RecyclerView recyclerView, int dx, int dy) {super.onScrolled(recyclerView, dx, dy);// …

關于深度學習,這些知識點你需要了解一下

深度學習概述 o 受限玻爾茲曼機和深度信念網絡 o Dropout o 處理不平衡的技巧 o SMOTE&#xff1a;合成少數過采樣技術 o 神經網絡中對成本敏感的學習 深度學習概述 在2006年之前&#xff0c;訓練深度監督前饋神經網絡總是失敗的&#xff0c;其主要原因都是導致…

發送不同類型的ActivityFeed

上一篇文章講到了如何使用最新的Graph API來給一個用戶發送一個簡單的 Activity Feed。我們這篇文章來詳細講一下發送三種不同類型的消息。 發送 Chat 相關的 Activity Notification API 為 POST https://graph.microsoft.com/beta/chats/{chat-id}/sendActivityNotification…

git add * 提示warning: LF will be replaced by CRLF in 解決辦法

在使用git的時候&#xff0c;每次執行 $ git add * 都會提示這樣一個警告消息&#xff1a; 雖然說沒有什么影響吧。 不過就是覺得太礙眼了&#xff0c; 按照這樣設置就沒有問題了: git config core.autocrlf false 這樣設置git的配置后在執行add操作就沒有問題了。 奮斗的年紀你…

git 放棄本地修改,強制拉取更新

開發時&#xff0c;對于本地的項目中修改不做保存操作&#xff08;或代碼改崩&#xff09;&#xff0c;可以用到Git pull的強制覆蓋&#xff0c;具體代碼如下&#xff1a; git fetch --all git reset --hard origin/master git pull //可以省略 git fetch 指令是下載遠程倉庫最…

發送ActivityFeed的隱藏功能

前兩篇文章介紹了如何發送 activity notification&#xff0c;這篇文章主要介紹兩個隱藏功能&#xff0c;實際上所謂的隱藏功能是指大家在閱讀官方文檔是會忽略的兩個點&#xff0c;但是實際上也是很實用的兩個功能點。 text 類型的 topic 之前文章中提到我們的 activity not…

Dispatch Queue 之 Invoke 當前隊列

&#xfffc; 轉載于:https://www.cnblogs.com/huahuahu/p/dispatch-queue-zhi-invoke-dang-qian-dui-lie.html