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