前幾天小編在講Servlet時(shí),為了靈活地使用同一個(gè)Servlet來處理對同一張表的業(yè)務(wù)操作請求,我給學(xué)生講解了BaseServlet工具類的封裝,基本實(shí)現(xiàn)思路有如下幾個(gè)步驟。
一. 反射封裝BaseServlet工具類
使用反射封裝BaseServlet工具類,無論是哪個(gè)Servlet接收到請求,都由該類完成請求分發(fā)。因此該類的主要作用就是通過反射機(jī)制,確定我們請求的到底是哪個(gè)Servlet的哪個(gè)方法。
/*
* BaseServlet 獲取客戶端請求的是哪個(gè)servlet的哪個(gè)方法
* */
public class BaseServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//獲取客戶端發(fā)來請求的標(biāo)識(shí):即要執(zhí)行的方法名
String method = req.getParameter("method");
//獲取方法屬于哪個(gè)Servlet類
Class<? extends BaseServlet> clazz = this.getClass();
//通過類字節(jié)碼對象獲取要執(zhí)行方法的對象
try {
Method mh = clazz.getMethod(method, HttpServletRequest.class, HttpServletResponse.class);
//執(zhí)行方法
mh.invoke(this,req,resp); // this.insert(req,resp);
} catch (Exception e) {
e.printStackTrace();
}
}
}
二. 繼承BaseServlet父類
以后再創(chuàng)建Servlet時(shí),我們要使任意一個(gè)Servlet類,不再直接繼承HttpServlet,而是要繼承統(tǒng)一的BaseServlet,完成請求的分發(fā)管理,例如:
三. 異常展現(xiàn)
然而有個(gè)別同學(xué)在按照上述思路自己編寫代碼時(shí),卻遇到了下面的NoSuchMethodException異常。他排查許久未果,于是就來找小編幫他解決。
四. 異常原因
起初,小編以為是學(xué)生從客戶端發(fā)出請求時(shí),未攜帶執(zhí)行方法的標(biāo)識(shí)或攜帶的方法標(biāo)識(shí)與實(shí)際方法名不匹配,從而導(dǎo)致通過反射機(jī)制獲取方法對象時(shí)報(bào)錯(cuò)。因?yàn)槲覀冎?,在通過Methodmh=clazz.getMethod(method,HttpServletRequest.class,HttpServletResponse.class)獲取Method對象時(shí),必須保證方法名、參數(shù)匹配,才能找到指定的方法,否則就會(huì)出現(xiàn)此類異常。
但經(jīng)過排查,發(fā)現(xiàn)并不是以上原因,該學(xué)生的代碼如下:
@WebServlet("/stuinfo")
public class StuinfoServlet extends BaseServlet {
//創(chuàng)建serivce層對象
private StuinfoService ss = new StuinfoSerivceImpl();
//查詢方法
private void findAll(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//調(diào)用service層查詢方法
List<Stuinfo> list = ss.findAll();
//將list集合數(shù)據(jù)保存到域?qū)ο笾?/span>
req.setAttribute("stuList",list);
//跳轉(zhuǎn)到主頁面
req.getRequestDispatcher("index.jsp").forward(req,resp);
}
}
五. 異常解決
可能你也一眼就看到了,上述代碼中,查詢方法使用的是private修飾符,而私有成員在該類的外部是不能被訪問的!因此我們在利用反射,通過getMethod()方法獲取Method對象時(shí)就會(huì)出現(xiàn)NoSuchMethodException異常。所以現(xiàn)在的解決辦法,你是不是立刻就明朗了,我們直接將private改成public就可以了。
六. 暴力反射
上面的問題是解決了,但大家還要知道,反射機(jī)制中還有一種叫暴力反射,聽起來是不是很厲害!!利用暴力反射,即使被private修飾也可以進(jìn)行正常的操作。
小編在這里給大家再補(bǔ)上一刀,反射里的Constructor、Field、Method三個(gè)類都有g(shù)etDeclaredXxx方法(這里的Xxx表示Constructor、Field、Method),該方法可以不受權(quán)限控制,就能夠獲取到類中的這些成員信息。如果我們想要使用私有的構(gòu)造函數(shù)、字段、方法,則會(huì)自動(dòng)訪問類的isAccessable,其默認(rèn)值是false,表示在訪問成員時(shí)需要安全檢查,如果發(fā)現(xiàn)是私有的則不允許訪問。所以,如果我們想要訪問類中的私有成員時(shí),需要調(diào)用setAccessible(boolean flag)方法,將其改為true。這樣,我們就可以對類中的私有成員進(jìn)行操作了。