基本Servlet的封装,实现可以通过继承一个BaseServlet,然后不需要重复定义多个Servlet,只需要在继承的这个类里面通过实现方法去访问这个Servlet

优化Servlet

减少Servlet 的数量,现在是一个功能一 个Servlet,将其优化为-一个模块一个Servlet,相当于在数据库中一张表对应一个Servlet,在Servlet中提供不同的方法,完成用户的请求。

image-20210905164351190

为什么这些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);
    }

}
  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

?abc!

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值