监控部分反馈异常,生产系统日志文件竟然木有了(最后一次版本发布后,再也无日志文件生成)。
问题排查步骤:
1. 检查logback配置文件
日志生成目录一切正常
应该服务器上磁盘空间未满、操作权限没有问题。
pass
2. 检查应用中logback对象是否加载成功
在web.xml中,通过context-param指定日志文件路径异常。
问题查找出来了。
错误的配置如下:
<context-param> <param-name>logbackConfigLocation</param-name> <param-value>file:/opt/pay/config/masopen/logback_masopen.xml</param-value> <param-name>webAppRootKey</param-name> <param-value>masopen</param-value> </context-param> <context-param> <param-name>contextConfigLocation</param-name> <param-value> classpath:/META-INF/spring/masopen_service.xml classpath:/META-INF/spring/masopen_task.xml classpath:/META-INF/spring/masopen_api_cxf.xml </param-value> </context-param>
很明显的错误,在<context-param>中竟然定义了2组<param-name>和<param-value>,自然以最下面定义的那组为准。
这种低级错误,竟然在代码评审中没有审查出来。
知识扩展:
1. context-param定义
<context-param> <param-name>webAppRootKey</param-name> <param-value>masopen</param-value> </context-param>
2. context-param初始化过程
初始化过程: 在启动Web项目时,容器(比如Tomcat)会读web.xml配置文件中的两个节点<listener>和<contex-param>。 接着容器会创建一个ServletContext(上下文),应用范围内即整个WEB项目都能使用这个上下文。 接着容器会将读取到<context-param>转化为键值对,并交给ServletContext。 容器创建<listener></listener>中的类实例,即创建监听(备注:listener定义的类可以是自定义的类但必须需要继承ServletContextListener)。 在监听的类中会有一个contextInitialized(ServletContextEvent event)初始化方法,在这个方法中可以通过event.getServletContext().getInitParameter("contextConfigLocation") 来得到context-param 设定的值。在这个类中还必须有一个contextDestroyed(ServletContextEvent event) 销毁方法.用于关闭应用前释放资源,比如说数据库连接的关闭。 得到这个context-param的值之后,你就可以做一些操作了.注意,这个时候你的WEB项目还没有完全启动完成.这个动作会比所有的Servlet都要早。
3. context-param如何使用
页面中
${initParam.contextConfigLocation}
Servlet中
String paramValue=getServletContext().getInitParameter("contextConfigLocation")