一、关于项目中使用struts2 报java.lang.NoSuchFieldException: resourceEntries 的问题。java
当咱们使用tomcat 7.0.64 及如下的版本时是不会出问题的。当咱们在使用tomcat 7.0.65时会报这个问题,网上看到不少解答都是说什么tomcat 8之类的。apache
实际出现问题在tomcat 7.0.65 就开始了 。出现这个问题的根源是由于。在64到65 的时候,org\apache\catalina\loader\WebappClassLoader.java 这个类进行了抽象,抽象出了org\apache\catalina\loader\WebappClassLoaderBase.java.缓存
二、在咱们用的struts2 的xwork 包内 \com\opensymphony\xwork2\util\LocalizedTextUtil.java内的清除tomcat 缓存的地方有问题。问题就初夏这个clearMap。java自带的getDeclaredField 是取不到父类的属性的。因此这里会一直报错。tomcat
最新版的这两个方法:app
private static void clearTomcatCache() { ClassLoader loader = getCurrentThreadContextClassLoader(); // no need for compilation here. Class cl = loader.getClass(); try { if ("org.apache.catalina.loader.WebappClassLoader".equals(cl.getName())) { clearMap(cl, loader, TOMCAT_RESOURCE_ENTRIES_FIELD); } else { if (LOG.isDebugEnabled()) { LOG.debug("class loader " + cl.getName() + " is not tomcat loader."); } } } catch (NoSuchFieldException nsfe) { if ("org.apache.catalina.loader.WebappClassLoaderBase".equals(cl.getSuperclass().getName())) { if (LOG.isDebugEnabled()) { LOG.debug("Base class #0 doesn't contain '#1' field, trying with parent!", nsfe, cl.getName(), TOMCAT_RESOURCE_ENTRIES_FIELD); } try { clearMap(cl.getSuperclass(), loader, TOMCAT_RESOURCE_ENTRIES_FIELD); } catch (Exception e) { if (LOG.isWarnEnabled()) { LOG.warn("Couldn't clear tomcat cache using #0", e, cl.getSuperclass().getName()); } } } } catch (Exception e) { if (LOG.isWarnEnabled()) { LOG.warn("Couldn't clear tomcat cache", e, cl.getName()); } } } private static void clearMap(Class cl, Object obj, String name) throws NoSuchFieldException, IllegalAccessException, NoSuchMethodException, InvocationTargetException { Field field = cl.getDeclaredField(name); field.setAccessible(true); Object cache = field.get(obj); synchronized (cache) { Class ccl = cache.getClass(); Method clearMethod = ccl.getMethod("clear"); clearMethod.invoke(cache); } }
三、遇到这个问题的时候,咱们每每在处理老项目,不少项目是由于扫描除了漏洞而不得不使用新版本的tomcat。咱们能够选择升级struts2的版本,但是版本的升级会带来不少意想不到的问题。摆在咱们面前三条路①、用最新版本的struts2.②、拿出这个类,从新一下这两个方法。③、更改一下这两个方法,再放到jar包里面。.net
参考:debug
apache commons lang3包下的FieldUtils.getAllFields()能够获取类和父类的全部(public、protected、default、private)属性。code
https://blog.csdn.net/wangjun5159/article/details/79289244blog