首页 > 代码库 > struts2: config-browser-plugin 与 convention-plugin 学习
struts2: config-browser-plugin 与 convention-plugin 学习
struts2被很多新手诟病的一个地方在于“配置过于复杂”,相信不少初学者因为这个直接改投Spring-MVC了。convention-plugin、 config-browser-plugin这二个插件的出现,很大程度改善了这个囧境。
简言之:convention-plugin采用"约定大于配置”的思想,只要我们遵守约定,完全可以少写配置甚至不写配置;而config-browser-plugin则用于方便的浏览项目中的所有action及其与jsp view的映射。这二个插件结合起来学习,能很方便的搞定struts2中各种复杂的action-view映射需求。
一、config-browser-plugin使用
1 <dependency>2 <groupId>org.apache.struts</groupId>3 <artifactId>struts2-config-browser-plugin</artifactId>4 <version>2.3.16</version>5 </dependency>
maven项目的pom.xml中加上这个即可,运行后,浏览 http://localhost:8080/{你的项目名称}/config-browser/ 即可看到当前项目中的所有action
注:以下内容中,凡有url的地方,项目名称假设为struts2-helloworld
如果跑不起来,检查服务器应用WEB-INF/lib/下是否有struts2-config-browser-plugin-2.3.16.jar 这个文件
二、convention-plugin 使用
1 <dependency>2 <groupId>org.apache.struts</groupId>3 <artifactId>struts2-convention-plugin</artifactId>4 <version>2.3.16</version>5 </dependency>
pom.xml中加上这个后,可以把struts.xml配置文件给干掉了(或者改个名),部署App,如果启动正常,则表示环境ok。如果提示缺少asm 啥类,检查下面这几个依赖项,是否也加进来了
1 <dependency> 2 <groupId>asm</groupId> 3 <artifactId>asm</artifactId> 4 <version>3.3.1</version> 5 </dependency> 6 7 <dependency> 8 <groupId>asm</groupId> 9 <artifactId>asm</artifactId>10 <version>3.3.1</version>11 </dependency>12 13 <dependency>14 <groupId>asm</groupId>15 <artifactId>asm-commons</artifactId>16 <version>3.3.1</version>17 </dependency>18 19 <dependency>20 <groupId>asm</groupId>21 <artifactId>asm-tree</artifactId>22 <version>3.3.1</version>23 </dependency>
2.1 零action的view
convention-plugin约定所有的jsp view都放在WEB-INF/content目录下,在这个目录下先随便放一个名为"no-action.jsp"的jsp文件,里面随便写点啥
浏览 http://localhost:8080/struts2-helloworld/no-action
即:即使没有对应的Action类,struts2也能按约定正常展现页面。(当然,这只是开胃小菜,真正应用中,除了做一些纯静态的页面原型之外,大部分场景,背后还是要有Action类来支撑的)
2.2 常规映射
建一个HelloWorld.action类
1 package com.cnblogs.yjmyzz.action; 2 3 import org.apache.struts2.convention.annotation.Action; 4 import org.apache.struts2.convention.annotation.Namespace; 5 import org.apache.struts2.convention.annotation.Result; 6 7 @Namespace("/home") 8 public class HelloWorldAction extends BaseAction { 9 10 private static final long serialVersionUID = -8827776224243873974L;11 12 private String message;13 14 @Action("hello-world")15 public String execute() throws Exception {16 return SUCCESS;17 }18 19 @Action(value = "http://www.mamicode.com/say-hi", results = { @Result(name = "success", location = "hello-world.jsp") })20 public String sayHi() throws Exception {21 message = "welcome to SSH!";22 return SUCCESS;23 }24 25 public String getMessage() {26 return message;27 }28 29 public void setMessage(String message) {30 this.message = message;31 }32 33 }
解释一下:
第7行,在整个Action类上使用了@Namespace("/home"),表示整个Action最终浏览的url,是以 http://localhost:8080/{你的项目名称}/home/ 打头
第14行,通过注解@Action("hello-world"),把默认的/home/index.action路径,改成了 /home/hello-world
至于execute方法,返回success后,对应的是哪个jsp文件,这个不用死记,通过config-browser-plugin看下便知
即:execute方法返回input/error/success中的任何一个,都会映射到/WEB-INF/content/home/hello-world.jsp 这个文件上
20行sayHI()方法上的注解有点意思,@Action(value = "http://www.mamicode.com/say-hi", results = { @Result(name = "success", location = "hello-world.jsp") }),默认情况下,如果不指定location,返回success时,它应该对应 /WEB-INF/content/home/say-hi.jsp这个文件,但是通过location值,变成了hello-world.jsp,即与/home/hello-world共用同一个view.
3、拦截器问题
上一篇学习了通过拦截器来处理异常,采用convention插件后,会发现拦截器不起作用(struts.xml中配置了也一样)
1 <?xml version="1.0" encoding="UTF-8" ?> 2 <!DOCTYPE struts PUBLIC 3 "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN" 4 "http://struts.apache.org/dtds/struts-2.3.dtd"> 5 6 <struts> 7 8 <constant name="struts.enable.DynamicMethodInvocation" value="false" /> 9 <constant name="struts.devMode" value="false" />10 11 <package name="default" namespace="/" extends="struts-default">12 13 <interceptors>14 <interceptor name="myinterceptor"15 class="com.cnblogs.yjmyzz.Interceptor.ExceptionInterceptor">16 </interceptor>17 18 <interceptor-stack name="myStack">19 <interceptor-ref name="myinterceptor" />20 </interceptor-stack>21 </interceptors>22 23 <default-interceptor-ref name="myStack" />24 <default-action-ref name="index" />25 26 <global-results>27 <result name="error">/WEB-INF/common/error.jsp</result>28 </global-results>29 30 <global-exception-mappings>31 <exception-mapping exception="java.lang.Exception"32 result="error" />33 </global-exception-mappings>34 35 <action name="index">36 <result type="redirectAction">37 <param name="actionName">hello-world</param>38 <param name="namespace">/home</param>39 </result>40 </action>41 42 </package>43 44 <!-- 因为有Convention-plugin,就不再需要手动写action-view的映射规则了 -->45 <!-- <include file="struts-home.xml" />46 <include file="struts-mytatis.xml" /> -->47 48 </struts>
原因在于convention-plugin使用后,所有Action不再继承自默认defaultStack对应的package,为了解决这个问题,建议所有Action都继承自一个自定义的BaseAction类,然后在BaseAction上加上@ParentPackage("default"),让所有Action强制继承自struts.xml中定义的default package
1 package com.cnblogs.yjmyzz.action; 2 3 import org.apache.struts2.convention.annotation.ParentPackage; 4 import org.slf4j.Logger; 5 import org.slf4j.LoggerFactory; 6 7 import com.opensymphony.xwork2.ActionSupport; 8 9 @ParentPackage("default")10 public class BaseAction extends ActionSupport {11 12 private static final long serialVersionUID = -4320398837758540242L;13 protected Logger logger = LoggerFactory.getLogger(this.getClass());14 15 }
基本用法就是这些,如果不清楚Action最终出来的url是啥,或者不清楚某个url最终会对应到哪个jsp文件,无需死记规则,只要善用config-browser-plugin,大多数下不用查阅文档即可快速解决问题。
struts2: config-browser-plugin 与 convention-plugin 学习