优化Servlet
减少Servlet 的数量,现在是一个功能一 个Servlet,将其优化为-一个模块一个Servlet,相当于在数据库中一张表对应一个Servlet,在Servlet中提供不同的方法,完成用户的请求。
为什么这些Servlet必须每一个Servlet都要写?
-
因为Servlet继承了HttpServlet,而在HttpServlet里面根据请求方式的不同,完成doPost和doGet方法的分发,
-
HttpServlet里面有一个方法,名为service方法,在这个方法里面完成的就是根据不同的请求方式(7种),完成了对方法的分发,然后我们通过去重写doget和doPost相关的方法完成真正的用户请求的一个处理
然后通过自定义的方法去实现方法的分发
-
我们不能去修改HttpServlet的代码,因为他是接口的规范,是sun公司提供的方法,我们不能去修改
-
我们可以单独自己去写一个Servlet(BaseServlet),将自己定义的Servlet都去继承这个Servlet(BaseServlet),在BaseServlet里面完成方法的分发
-
让BaseServlet去实现Servlet的规范,继承HttpServlet,这样的话自己的Servlet因为继承了BaseServlet,所以可以被看成是一个Servlet,可以被浏览器通过URL去访问到
在BaseServlet里面通过重写service()方法去完成方法的分发
- 希望可以实现,主要是继承了BaseServle,那么该方法可以通过方法名,就能够去访问
- BaseServlet不需要被访问到
- 当自己定义的其他Servlet继承了BaseServlet,那么通过访问自己定义的Servlet,也不会报错,并且会成功访问到BaseServlet里面重写的service方法 ,所以可以在service里面完成方法的分发
- req.getRequestURI():获取请求路径
- substring:通过这个去截取获得的URI里面传递过来的方法名称
- // /travel/user/add
- 在继承BaseServlet里面的所有方法,的参数和返回值都是一样的,除了方法的名称不一样之外,其他的都一样
- 通过this可以获取谁调用这个方法的字节码文件对象this.getClass().getDeclaredMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);获取对应方法的对象
- 过后就可以执行对象,通过invoke完成方法的分发
具体代码实现
BaseServlet
import java.io.IOException;
import java.lang.reflect.InvocationTargetException;
import javax.servlet.ServletException;
import javax.servlet.annotation.WebServlet;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.lang.reflect.Method;
public class BaseServlet extends HttpServlet {
@Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
//System.out.println("baseServlet的service方法被执行了...");
//完成方法分发
//1.获取请求路径
String uri = req.getRequestURI(); // /travel/user/add
System.out.println("请求uri:"+uri);// /travel/user/add
//2.获取方法名称
String methodName = uri.substring(uri.lastIndexOf('/') + 1);
System.out.println("方法名称:"+methodName);
//3.获取方法对象Method
//谁调用我?我代表谁
System.out.println(this);//UserServlet的对象cn.itcast.travel.web.servlet.UserServlet@4903d97e,对应的谁调用这个方法就能够获取这个方法的对象,比如在这里userServlet方法继承的是BaseServlet,所有打印出的是userServlet的对象
try {
//
//获取方法
//忽略访问修饰符,暴力破解,如果把权限修饰符改为public,还是不能实现访问,那么久使用暴力破解,这里的public是在继承这个类里面的方法的修饰符
Method method = this.getClass().getDeclaredMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
//尽量使用这个,比较好的
// Method method = this.getClass().getMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
//4.执行方法
//暴力反射
//method.setAccessible(true);
method.invoke(this,req,resp);
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
}
}
/**
* 直接将传入的对象序列化为json,并且写回客户端
* @param obj
*/
public void writeValue(Object obj,HttpServletResponse response) throws IOException {
ObjectMapper mapper = new ObjectMapper();
response.setContentType("application/json;charset=utf-8");
//这个是将obj里面的内容(对象)通过response传输浏览器
mapper.writeValue(response.getOutputStream(),obj);
}
/**
* 将传入的对象序列化为json,返回一个字符串的JSON,给调用者
* @param obj
* @return
*/
public String writeValueAsString(Object obj) throws JsonProcessingException {
ObjectMapper mapper = new ObjectMapper();
//返回一个JSON
return mapper.writeValueAsString(obj);
}
}