反射和動態加載bean

最近我們部門有個小項目,用來管理這個公司所有項目用到的代碼表,例如國家代碼、行政區劃代碼等。這個項目的功能其實很少,就是簡單的修改、查詢、新增和邏輯刪除。但是爲每張表都寫一套增刪改查的頁面和一套service,工作量巨大,且維護很困難。我們發現各個表的業務其實都很類似,如果能寫一套通用的service代碼,在web層根據表名動態調用,在通用的jsp上生成想要的數據,那就只需要寫一套代碼就可以完成所有代碼表的操作了。

完成這個要求第一個想到的當然是用反射啊,通過表名反射生成相應的QO、DAO,通過調用相應的dao方法,來實現相關功能。但是當我通過反射實例出dao,再執行相應的方法時,方法內一直報錯。通過查詢資料,可以通過方法得到已經被Spring託管的bean( XX.getBean("xxx") ),於是我改成根據表名動態獲取dao,通過反射找到相應方法,執行相應方法,問題就迎刃而解了。

首先是獲取被Spring託管類的工具類

strip

1importorg.slf4j.Logger;2importorg.slf4j.LoggerFactory;3importorg.springframework.beans.factory.DisposableBean;4importorg.springframework.context.ApplicationContext;5importorg.springframework.context.ApplicationContextAware;6importorg.springframework.stereotype.Service;78/**9* 以靜態變量保存Spring ApplicationContext, 可在任何代碼任何地方任何時候中取出ApplicaitonContext.10*/11@Service12publicclassSpringContextHolderimplementsApplicationContextAware, DisposableBean {1314privatestaticApplicationContext applicationContext =null;1516privatestaticLogger logger = LoggerFactory.getLogger(SpringContextHolder.class);1718/**19* 實現ApplicationContextAware接口, 注入Context到靜態變量中.20*/21publicvoidsetApplicationContext(ApplicationContext applicationContext) {22logger.debug("注入ApplicationContext到SpringContextHolder:" +applicationContext);2324if(SpringContextHolder.applicationContext !=null) {25logger.warn("SpringContextHolder中的ApplicationContext被覆蓋, 原有ApplicationContext爲:"26+SpringContextHolder.applicationContext);27}2829SpringContextHolder.applicationContext = applicationContext;//NOSONAR30}3132/**33* 實現DisposableBean接口,在Context關閉時清理靜態變量.34*/35@Override36publicvoiddestroy()throwsException {37SpringContextHolder.clear();38}3940/**41* 取得存儲在靜態變量中的ApplicationContext.42*/43publicstaticApplicationContext getApplicationContext() {44assertContextInjected();45returnapplicationContext;46}4748/**49* 從靜態變量applicationContext中取得Bean, 自動轉型爲所賦值對象的類型.50*/51@SuppressWarnings("unchecked")52publicstaticT getBean(String name) {53assertContextInjected();54return(T) applicationContext.getBean(name);55}5657/**58* 從靜態變量applicationContext中取得Bean, 自動轉型爲所賦值對象的類型.59*/60publicstatic T getBean(Cla***equiredType) {61assertContextInjected();62returnapplicationContext.getBean(requiredType);63}6465/**66* 清除SpringContextHolder中的ApplicationContext爲Null.67*/68publicstaticvoidclear() {69logger.debug("清除SpringContextHolder中的ApplicationContext:" +applicationContext);70applicationContext =null;71}7273/**74* 檢查ApplicationContext不爲空.75*/76privatestaticvoidassertContextInjected() {77if(applicationContext ==null) {78thrownewIllegalStateException("applicaitonContext未注入,請在applicationContext.xml中定義SpringContextHolder");79}80}81}

strip

之後是service層代碼

strip

1publicinterfaceCommonService {23/**4* 分頁查詢5*@paramobject  表名6*@returnPage<>  列表7*/8Object findPageList(String name);9}

strip

strip

1/**2*3* 通用service4*/5@Service("commonService")6publicclassCommonServiceImplimplementsCommonService {7/**8* 分頁查詢9*@paramobject  表名10*@returnPage<>  列表11*/12publicObject findPageList(String name) {13try{14//獲取QO15Class clQO = Class.forName("org.xxx.domain.qo."+name+"QO");16Object objectQO =clQO.newInstance();17//設置未刪除18Field deltIndc = clQO.getDeclaredField("deltIndc");19deltIndc.setAccessible(true);20deltIndc.set(objectQO, "0");21//首字母小寫    拼成xxxxDao22String nameDao =23(newStringBuilder()).append(24Character.toLowerCase(name.charAt(0))).append(name.substring(1)).toString()25+"Dao";26//獲取被Spring託管的dao層實例27Object objDao =SpringContextHolder.getBean(nameDao);28Method[] ms =objDao.getClass().getMethods();29for(Method m:ms){30//找到find方法 執行31if("find".equals(m.getName())){32returnm.invoke(objDao,newObject[]{objectQO});33}34}35}catch(Exception e) {36e.printStackTrace();37}38returnnull;39}40}

strip

controller

strip

1@Controller2publicclassCommonAction {345@Autowired(required =false)6privateCommonService commonService;78@RequestMapping("getPage")9public@ResponseBody Object getPage(String name) {10//"PltmPubNews"11returncommonService.findPageList(name);12}1314}

strip

這樣就可以根據表名動態查詢了。

歡迎加入技術QQ羣:364595326



發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章