struts2被很多新手诟病的一个地方在于“配置过于复杂”,相信不少初学者因为这个直接改投Spring-MVC了。convention-plugin、 config-browser-plugin这二个插件的出现,很大程度改善了这个囧境。

简言之:convention-plugin采用"约定大于配置”的思想,只要我们遵守约定,完全可以少写配置甚至不写配置;而config-browser-plugin则用于方便的浏览项目中的所有action及其与jsp view的映射。这二个插件结合起来学习,能很方便的搞定struts2中各种复杂的action-view映射需求。

一、config-browser-plugin使用

 <dependency>
<groupId>org.apache.struts</groupId>
<artifactId>struts2-config-browser-plugin</artifactId>
<version>2.3.16</version>
</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 使用

 <dependency>
<groupId>org.apache.struts</groupId>
<artifactId>struts2-convention-plugin</artifactId>
<version>2.3.16</version>
</dependency>

pom.xml中加上这个后,可以把struts.xml配置文件给干掉了(或者改个名),部署App,如果启动正常,则表示环境ok。如果提示缺少asm 啥类,检查下面这几个依赖项,是否也加进来了

 <dependency>
<groupId>asm</groupId>
<artifactId>asm</artifactId>
<version>3.3.1</version>
</dependency> <dependency>
<groupId>asm</groupId>
<artifactId>asm</artifactId>
<version>3.3.1</version>
</dependency> <dependency>
<groupId>asm</groupId>
<artifactId>asm-commons</artifactId>
<version>3.3.1</version>
</dependency> <dependency>
<groupId>asm</groupId>
<artifactId>asm-tree</artifactId>
<version>3.3.1</version>
</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类

 package com.cnblogs.yjmyzz.action;

 import org.apache.struts2.convention.annotation.Action;
import org.apache.struts2.convention.annotation.Namespace;
import org.apache.struts2.convention.annotation.Result; @Namespace("/home")
public class HelloWorldAction extends BaseAction { private static final long serialVersionUID = -8827776224243873974L; private String message; @Action("hello-world")
public String execute() throws Exception {
return SUCCESS;
} @Action(value = "say-hi", results = { @Result(name = "success", location = "hello-world.jsp") })
public String sayHi() throws Exception {
message = "welcome to SSH!";
return SUCCESS;
} public String getMessage() {
return message;
} public void setMessage(String message) {
this.message = message;
} }

解释一下:

第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 = "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中配置了也一样)

 <?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
"http://struts.apache.org/dtds/struts-2.3.dtd"> <struts> <constant name="struts.enable.DynamicMethodInvocation" value="false" />
<constant name="struts.devMode" value="false" /> <package name="default" namespace="/" extends="struts-default"> <interceptors>
<interceptor name="myinterceptor"
class="com.cnblogs.yjmyzz.Interceptor.ExceptionInterceptor">
</interceptor> <interceptor-stack name="myStack">
<interceptor-ref name="myinterceptor" />
</interceptor-stack>
</interceptors> <default-interceptor-ref name="myStack" />
<default-action-ref name="index" /> <global-results>
<result name="error">/WEB-INF/common/error.jsp</result>
</global-results> <global-exception-mappings>
<exception-mapping exception="java.lang.Exception"
result="error" />
</global-exception-mappings> <action name="index">
<result type="redirectAction">
<param name="actionName">hello-world</param>
<param name="namespace">/home</param>
</result>
</action> </package> <!-- 因为有Convention-plugin,就不再需要手动写action-view的映射规则了 -->
<!-- <include file="struts-home.xml" />
<include file="struts-mytatis.xml" /> --> </struts>

原因在于convention-plugin使用后,所有Action不再继承自默认defaultStack对应的package,为了解决这个问题,建议所有Action都继承自一个自定义的BaseAction类,然后在BaseAction上加上@ParentPackage("default"),让所有Action强制继承自struts.xml中定义的default package

 package com.cnblogs.yjmyzz.action;

 import org.apache.struts2.convention.annotation.ParentPackage;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory; import com.opensymphony.xwork2.ActionSupport; @ParentPackage("default")
public class BaseAction extends ActionSupport { private static final long serialVersionUID = -4320398837758540242L;
protected Logger logger = LoggerFactory.getLogger(this.getClass()); }

基本用法就是这些,如果不清楚Action最终出来的url是啥,或者不清楚某个url最终会对应到哪个jsp文件,无需死记规则,只要善用config-browser-plugin,大多数下不用查阅文档即可快速解决问题。

最新文章

  1. 【原创】PageAdminCMS 前台SQL注入漏洞(2)
  2. RabbitMQ详解
  3. I/O复用
  4. Android EditText 获得输入焦点 以及requestfocus()失效的问题
  5. 推荐一个大文件查找工具---WizTree
  6. SSL 通信原理及Tomcat SSL 配置
  7. jqgrid在页面出来竖型滚动条自动调整列宽
  8. last_9t&#39;s_ramsey
  9. Bzoj 1976: [BeiJing2010组队]能量魔方 Cube 最小割,最大流
  10. Document原来可以这样来获取DOM
  11. 【Windows 8 Store App】学习二:ResourceLoader
  12. 初学Python(九)——函数
  13. PHP 两张图片合成一张图片(加水印)
  14. jstl标签库示例二
  15. Chapter 5 Blood Type——4
  16. C# 动态代码生成控件后其他事件不能获取该控件的值
  17. Java并发编程学习:线程安全与锁优化
  18. 图集内子图压缩及 ETC2 fallback选项的作用
  19. CPP之内存分配
  20. VSS2005源代码管理

热门文章

  1. RESTful API你怎么看?
  2. MapReduce二次排序
  3. linux性能监控工具
  4. 0014 Java学习笔记-集合-HashMap集合
  5. Navicat常用快捷键
  6. 简析一下SQL Server里面Fast_Forword 和 SRROLL 的区别
  7. [转]输出带颜色的shell
  8. react native windows开发环境搭建(二)
  9. RAID详解[RAID0/RAID1/RAID10/RAID5]
  10. mysqli,Fatal error