(1)问题介绍
修改物料,部分功能描述如下:
选择需要修改的物料,单击修改按钮,调用ShowModifyItemServlet进行一系列设置,并转发到物料修改页面item_modify.jsp。
在这个过程中,出现错误,打印堆栈如下:
四月 01, 2012 5:35:52 下午 org.apache.catalina.core.StandardWrapperValve invoke
严重: Servlet.service() for servlet [ShowModifyItemServlet] in context with path [/drp5.0] threw exception
java.lang.NullPointerException
at com.ys.drp.basedata.web.ShowModifyItemServlet.service(ShowModifyItemServlet.java:43)
at javax.servlet.http.HttpServlet.service(HttpServlet.java:722)
at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:305)……
(2)问题分析
由打印的堆栈错误可以看出:
抛出错误:Servlet.service() for servlet [ShowModifyItemServlet] in context with path [/drp5.0] threw exception
错误类型:java.lang.NullPointerException,空指针错误
出现错误的位置:com.ys.drp.basedata.web.ShowModifyItemServlet.service(ShowModifyItemServlet.java:43)
ShowModifyItemServlet代码如下:
public class ShowModifyItemServlet extends AbstractItemServlet {
private DataDictManager dataDictManager;
@Override
public void init() throws ServletException {
dataDictManager = DataDictManager.getInstance();
}
@Override
protected void service(HttpServletRequest request, HttpServletResponse response)
throws ServletException, IOException {
//取得物料代码
String itemNo = request.getParameter("itemNo");
//根据物料代码查询
。 Item item = itemManager.findItemById(itemNo); //在此处设置断点
//将物料信息设置到request中
request.setAttribute("item", item);
//...
//转发(传递数据)
request.getRequestDispatcher("/basedata/item_modify.jsp").forward(request, response);
}
}
在上面所示断点位置,设置断点,运行结果如下:
itemManager = null,找到了空指针的对象,堆栈中打印的空指针错误,便由此而来。
(3)改错
Ctrl+鼠标左键,进入itemManager,代码如下:
public abstract class AbstractItemServlet extends HttpServlet {
protected ItemManager itemManager;
@Override
public void init() throws ServletException {
BeanFactory beanFactory = (BeanFactory)this.getServletContext().getAttribute("beanFactory");
itemManager = (ItemManager)beanFactory.getServiceObject(ItemManager.class);
}
}
在抽象类的init()方法中创建itemManager。
再来看我们上面的代码,ShowModifyItemServlet继承抽象类,而且ShowModifyItemServlet也有自己的init()方法。因此,在运行时,子类的init()方法就把父类的init()方法给覆盖掉了,就出现了我们上面的情况。
修改子类的init()方法,加入super.init(),如下:
public void init() throws ServletException {
dataDictManager = DataDictManager.getInstance();
super.init();
}
这次的运行结果有值了:
(4)结果
继续运行,页面终于出来了。