作为servlet学习小白, 今天在编写好自己的servlet类之后, 通过启动tomcat来调试servlet, 结果发现在初始化时报错了, 以下是还原过程:
问题再现
已经启动好了tomcat, 启动正常, 通过url访问servlet时, 页面报500错误:
问题分析与解决
通过阅读报错信息: Error instantiating servlet class [com.amaker.servlet.HelloServlet],以及java.lang.NoSuchMethodException: com.amaker.servlet.HelloServlet.(), 认为是初始化方法不对, 但是查看了init方法没发现什么问题.
init方法:
// 初始化方法
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("init");
this.servletConfig = servletConfig;
this.servletContext = servletConfig.getServletContext();
}
突然, 联想到servlet的生命周期: 构造一个servlet对象->初始化init方法->service方法->destroy方法. 故问题还有可能出在构造方法上, 于是查看了servlet的构造方法, 编写的是一个默认访问权限的构造方法, 尝试将访问权限改成public后, 问题解决.
原先的构造器:
HelloServlet(){
System.out.println("construcotors");
}
修改了访问权限后的的构造器
public HelloServlet(){
System.out.println("construcotors");
}
结论
在自己编写的servlet实现类中, 构造器如果不是public级别的, tomcat容器就不能够实例化这个servlet实现类, 导致无法使用到其中的方法而报500错误.
所以, 自己编写的servlet实现类, 可以不写构造器, 默认就是public级别的, 如果写了一定显式声明public.
附上原报错的全量代码
public class HelloServlet implements Servlet{
private ServletConfig servletConfig;
private ServletContext servletContext;
HelloServlet(){
System.out.println("construcotors");
}
@Override
public void destroy() {
System.out.println("destroy");
}
@Override
public ServletConfig getServletConfig() {
System.out.println("getServletConfig");
return null;
}
@Override
public String getServletInfo() {
System.out.println("getServletInfo");
return null;
}
@Override
public void init(ServletConfig servletConfig) throws ServletException {
System.out.println("init");
this.servletConfig = servletConfig;
this.servletContext = servletConfig.getServletContext();
}
@Override
public void service(ServletRequest req, ServletResponse resp) throws ServletException, IOException {
System.out.println("service");
resp.getWriter().println("Hello Servlet");
}
}