在一个使用Struts2(2.3.20)的项目中,我想通过 应用程序启动时配置的操作(名称,类,命名空间,方法)。
我正在使用
供参考:我之前已经完成了bean和Struts注入的一些工作,所以不完全是新鲜的,但是我没有解决这里所说的问题。
任何关于如何获得这一点的指示都将受到赞赏。
阅读Andrea的回答我看到我需要解释我需要什么。
我正在为应用程序构建应用程序菜单构建器功能。我的计划是获取动作配置并构建一个菜单节点树"来自选定行动类和方法的注释中的信息。
我对配置浏览器中的代码的问题是Configuration
(xwork)似乎不在Struts组件之外可用。由于这是一个应用程序启动任务,因此它并不适合Struts' MVC组件模型。我想将菜单构建初始化放在ServletContextListener
。
此处的每个请求只是连接actionconfig< - >注释< - > my_custom_menu。从这里我可以生成一个菜单结构,从动作类和方法的注释提供。
public class ActionMenuBuilderListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent arg0) {
List<ActionCfg> actions = Struts.getConfiguredActions(); // thisi is where I'd like some help
for(ActionCfg action : actions) {
MenuAnnotation annotation = getAnnotationFromMethodOrClass(action);
if(annotation != null) {
addMenuItem(action, annotation);
}
}
}
}
这里ActionCfg
是Struts为操作配置返回的任何类,Struts.getConfiguredActions()
将是对Struts组件的一次或多次调用,addMenu(...)
是我向我的结构添加菜单项节点的地方。该结构后来成为JSP-s构建菜单的目标。
我不知道要编写多少代码。
对于完整性,我认为我将包括从中产生的内容。
首先,我通过这个插入Struts
ServletContextListnere
:
public class ActionMenuBuilderListener implements
ServletContextListener {
@Override
public void contextDestroyed(ServletContextEvent arg0) {
}
@Override
public void contextInitialized(ServletContextEvent event) {
ActionMenuDispatcherListener listener =
new ActionMenuDispatcherListener();
ServletContext context = event.getServletContext();
listener.setServletContext(context);
Dispatcher.addDispatcherListener(listener);
}
}
然后,我写了DispatcherListener
:
public class ActionMenuDispatcherListener implements DispatcherListener {
private ServletContext servletContext;
...
@Override
public void dispatcherInitialized(Dispatcher dispatcher) {
Map<String, PackageConfig> packages = dispatcher
.getConfigurationManager().getConfiguration()
.getPackageConfigs();
Map<String, Map<String, ActionConfig>> runtimeActionConfigs = dispatcher
.getConfigurationManager().getConfiguration()
.getRuntimeConfiguration().getActionConfigs();
for (String packageKey : runtimeActionConfigs.keySet()) {
Map<String, ActionConfig> actionConfigs = runtimeActionConfigs
.get(packageKey);
for (String actionKey : actionConfigs.keySet()) {
ActionConfig actionConfig = actionConfigs.get(actionKey);
PackageConfig packageConfig = packages.get(actionConfig
.getPackageName());
if (packageConfig != null) {
String actionName = actionConfig.getName();
String namespace = packageConfig.getNamespace();
try {
ActionMenu methodAnnotation = getMethodAnnotation(actionConfig);
if (methodAnnotation != null) {
String annotationInfo = methodAnnotation.value();
log.debug("[{}, {}, {}]", namespace, actionName,
annotationInfo);
}
} catch (ClassNotFoundException e) {
log.error("{}: {}", e.getClass().getSimpleName(),
e.getMessage());
}
}
}
}
}
protected ActionMenu getMethodAnnotation(ActionConfig actionConfig)
throws ClassNotFoundException {
String className = actionConfig.getClassName();
String methodName = actionConfig.getMethodName();
Class<?> actionClass = Class.forName(className);
try {
Method method = actionClass.getDeclaredMethod(methodName, null);
ActionMenu annotation = method.getAnnotation(ActionMenu.class);
return annotation;
} catch (NoSuchMethodException | SecurityException e) {
// log.error("{}: {}", e.getClass().getSimpleName(),
// e.getMessage());
}
return null;
}
}
以防万一其他人在考虑这些问题:)
答案 0 :(得分:5)
首先,您需要在加载和解析配置后挂钩应用程序初始化过程。其中一种方法是实施DispatcherListener
,您需要将其添加到Dispatcher
。您可以使用ServletContextListener#contextInitialized
方法执行此操作。
第二部分是获取动作配置。这很简单,因为Dispatcher
的实例作为参数传递给dispatcherInitialized
方法。要获取所有当前操作配置,请RuntimeConfiguration
保存Map<String, Map<String, ActionConfig>>
中的数据,其中第一个映射键是包命名空间,第二个映射键是操作名称,ActionConfig
包含有关操作的所有信息。由于您需要一个类名,请使用它的getClassName()
方法。
public class ActionMenuBuilderListener implements ServletContextListener,DispatcherListener {
@Override
public void contextInitialized(ServletContextEvent sce) {
Dispatcher.addDispatcherListener(this);
}
@Override
public void dispatcherInitialized(Dispatcher du) {
Map<String, Map<String, ActionConfig>> runtimeActionConfigs = du
.getConfigurationManager().getConfiguration().getRuntimeConfiguration()
.getActionConfigs();
}
// other methods
}
当然,不要忘记在web.xml中注册你的监听器。
答案 1 :(得分:3)
你可以为个人成长而建立这个东西,但要小心它已经存在。
它被称为Config Browser Plugin(struts2-config-browser-plugin-2.3.20.jar
)。
默认情况下,它包含在Maven原型中,您必须记住在投入生产之前将其删除。
导入后,可在以下网址找到:
// www.SERVER_NAME.com:8080/WEBAPP_NAME/config-browser/actionNames
它为您提供了您正在寻找的确切信息:操作,方法,结果,参数,映射等,它看起来像这样: