今天在工作中遇见一个代码优化常出错的BUG,自己整理下。
开始我开发的功能模块运行没问题,但后来进行嗲吗优化后就出错了查了好久才整出来。
先看下报的BUG 提示:
解释下错误:
Unable to instantiate Action, ....action.CertAction, defined for '...' in namespace '/...'null 原因是变量定义:
TbUser cUser = (TbUser) ActionContext.getContext().getSession().get(
"cuser");
关键是这里,这个cUser 成员变量使用了 action 的上下文"ServletActionContext.getContext()".
了解webwork中ServletDispatcher原理的人可能知道原因了, 因为ServletDispatcher接受客户端的HTTP请求,将JavaServlet的很多相关对象进行包装,再传给我们的XWork框架,由我 们的XWork 框架去解析我们的xwork.xml配置文件,根据配置文件的信息,创建对应的Action,组装并调用相应的拦截器,执行Action,返回执行结果。
因此每次客户端的请求都将调用ServletDispatcher 的 service()方法,
该方法执行顺序如下:
1、通过request请求取得action的命名空间
2、根据servlet请求的Path,解析出要调用该请求的Action的名字(actionName)
3、创建Action上下文(extraContext)
4、根据前面获得的namespace、actionName、extraContext,创建一个ActonProxy
5、执行proxy的execute()方法,根据Action执行返回的值去调用执行相应的Result(返回结果处理)的方法。
现在应该能明白,原来是Action对象实例在ActionContext对象实例之前创建的,所有这样取得ActionContext容器对象就有可能会返回null.
因此解决次问题方法很简单,就是最好不要在成员变量或者是构造方法里调用ActionContext,而把它放在其它方法里.
开始我开发的功能模块运行没问题,但后来进行嗲吗优化后就出错了查了好久才整出来。
先看下报的BUG 提示:
Unable to instantiate Action, markAction, defined for 'mark' in namespace '/'Error creating bean with name 'markAction' defined in file [D:\Program Files\Apache Software Foundation\Tomcat 6.0\webapps\OpinionPublicInternet_Norm\WEB-INF\classes\applicationContext-action.xml]: Instantiation of bean failed; nested exception is org.springframework.beans.BeanInstantiationException: Could not instantiate bean class [opinion.internet.hongming.net.action.MarkAction]: Constructor threw exception; nested exception is java.langNullPointerException - action - file:/D:/Program%20Files/Apache%20Software%20Foundation/Tomcat%206.0/webapps/OpinionPublicInternet_Norm/WEB-INF/classes/struts_xzs.xml:8:71
解释下错误:
Unable to instantiate Action, ....action.CertAction, defined for '...' in namespace '/...'null 原因是变量定义:
TbUser cUser = (TbUser) ActionContext.getContext().getSession().get(
"cuser");
关键是这里,这个cUser 成员变量使用了 action 的上下文"ServletActionContext.getContext()".
了解webwork中ServletDispatcher原理的人可能知道原因了, 因为ServletDispatcher接受客户端的HTTP请求,将JavaServlet的很多相关对象进行包装,再传给我们的XWork框架,由我 们的XWork 框架去解析我们的xwork.xml配置文件,根据配置文件的信息,创建对应的Action,组装并调用相应的拦截器,执行Action,返回执行结果。
因此每次客户端的请求都将调用ServletDispatcher 的 service()方法,
该方法执行顺序如下:
1、通过request请求取得action的命名空间
2、根据servlet请求的Path,解析出要调用该请求的Action的名字(actionName)
3、创建Action上下文(extraContext)
4、根据前面获得的namespace、actionName、extraContext,创建一个ActonProxy
5、执行proxy的execute()方法,根据Action执行返回的值去调用执行相应的Result(返回结果处理)的方法。
现在应该能明白,原来是Action对象实例在ActionContext对象实例之前创建的,所有这样取得ActionContext容器对象就有可能会返回null.
因此解决次问题方法很简单,就是最好不要在成员变量或者是构造方法里调用ActionContext,而把它放在其它方法里.