第一个struts案例及分析

时间:2021-06-04 22:07:15

软件中的框架,是一种半成品; 我们项目开发需要在框架的基础上进行!因为框架已经实现了一些功能,这样就可以提高开发效率!

Struts2 = struts1  +  xwork  (struts是基于MVC模式的框架)

struts2预先实现的功能:

  1、请求数据自动封装

  2、文件的上传

  3、国际化功能的简化

  4、数据效验

  ..........

Struts开发步骤:

1. web项目,引入struts - jar包

版本: 2.3

  commons-fileupload-1.2.2.jar   【文件上传相关包】

  commons-io-2.0.1.jar

  struts2-core-2.3.4.1.jar           【struts2核心功能包】

  xwork-core-2.3.4.1.jar           【Xwork核心包】

  ognl-3.0.5.jar                          【Ognl表达式功能支持表(类似于el表达式,但语法不同)】

  commons-lang3-3.1.jar          【struts对java.lang包的扩展】

  freemarker-2.3.19.jar            【struts的标签模板库jar文件】

  javassist-3.11.0.GA.jar           【struts对字节码的处理相关jar】

2. web.xml中,引入struts的核心功能(在项目的web.xml中引入过滤器),如下

 (struts的核心功能是通过滤器来完成初始化,Filter【init(服务器启动的时候执行)/doFilter(用户访问的时候执行)/destory(销毁执行)】,这个过滤器会过滤用户的请求,如果是以action类,该请求就会被转入struts2的框架中处理)

<!-- 引入struts核心过滤器 -->
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

这个class文件的路径下的StrutsPrepareAndExecuteFilter就是一个核心过滤器(注意: 使用的struts的版本不同,核心过滤器类是不一样的!)

3. 开发action

(action类—动作类,取代之前的servlet,即处理请求类,action中的业务方法,处理具体的请求,一般返回string,方法中不带有参数)

4. 配置action

src/struts.xml,里面定义了一些列的action,也指定了这些action的实现类,并定义了这些action的处理结构与资源视图的映射关系,如下:

<?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"> <!-- START SNIPPET: xworkSample -->
<struts>
<package name="gqx" extends="struts-default">
<!-- 定义了一个test的action,具体实现com.gqx.action.HelloAction类 -->
<action name="test" class="com.gqx.action.HelloAction" method="execute">
<!-- 配置action返回为success的时候,页面将跳转到success.jsp中 -->
<result name="success">/success.jsp</result>
</action>
</package> </struts> <!-- END SNIPPET: xworkSample -->

案例:

1、导入响应的jar包之后。配置xml文件(web.xml)

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" id="WebApp_ID" version="3.0">
<display-name>firstStruts</display-name> <!-- 引入struts核心过滤器 -->
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<welcome-file-list>
<welcome-file>index.html</welcome-file>
<welcome-file>index.htm</welcome-file>
<welcome-file>index.jsp</welcome-file>
<welcome-file>default.html</welcome-file>
<welcome-file>default.htm</welcome-file>
<welcome-file>default.jsp</welcome-file>
</welcome-file-list>
</web-app>

2、开发action,处理请求,通常实现actionSupport接口。

package com.gqx.action;

import com.opensymphony.xwork2.ActionSupport;

//开发action,处理请求
public class HelloAction extends ActionSupport{ //处理请求
@Override
public String execute() throws Exception {
// TODO Auto-generated method stub
System.out.println("访问到了action,正在处理请求。");
System.out.println("使用service方法");
return "success";
}
}

3、配置action,在src下写入struts.xml文件(和前面写的那个mystruct相似)

<?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"> <!-- START SNIPPET: xworkSample -->
<struts>
<package name="gqx" extends="struts-default">
<action name="test" class="com.gqx.action.HelloAction" method="execute">
<result name="success">/success.jsp</result>
</action>
</package> </struts> <!-- END SNIPPET: xworkSample -->

4、在success.jsp页面写入

<body>
This is success page. <br>
</body>

访问写的action(根据在struts.xml中配置的信息,根据action的name去访问即可得到相对应的信息),如下,访问http://localhost:8080/firstStruts/test,页面会自动跳转到success.jsp

第一个struts案例及分析


struts执行流程

一、服务器启动:

1. 加载项目web.xml

2. 创建Struts核心过滤器对象, 执行filter中的init(),里面有加载以下文件的方法

(1)    struts-default.xml,    核心功能的初始化(在struts2-core-2.3.4.1.jar目录下)

①、通过bean节点指定struts在运行的时候创建的对象类型   

<bean type="org.apache.struts2.components.UrlRenderer" name="struts" class="org.apache.struts2.components.ServletUrlRenderer"/>

②、指定struts包(用户写的package(struts.xml)一样要继承此包)

<package name="struts-default" abstract="true">
<result-types>
<result-type name="chain" class="com.opensymphony.xwork2.ActionChainResult"/>
.......
</result-types>
</package>

result-types:跳转的结果类型,在struts.xml中不写type类型默认是转发(dispatcher);

        [redirectAction:是指跳转(重定向)到另外一个action]

        [stream:文件下载的时候会用到]

③、定义了所有的拦截器(一共32个拦截器)

 <interceptors>
<interceptor name="alias" class="com.opensymphony.xwork2.interceptor.AliasInterceptor"/>
        .....
</interceptors>

在拦截器中为了拦截器引用方便,通过定义栈的方式引用拦截器,此时如果引用了栈,则所有的拦截器都会被使用,拦截器的执行顺序是按栈中的排列顺序执行的。

<interceptor-stack name="defaultStack">
<interceptor-ref name="exception"/>
<interceptor-ref name="alias"/>
......
</interceptor-stack>

④、在拦截器的后面就立刻引用了defaultStack,默认执行上面的18个拦截器栈(defaultStack栈)、默认执行的action

<default-interceptor-ref name="defaultStack"/>

<default-class-ref class="com.opensymphony.xwork2.ActionSupport" />

备注:拦截器功能与过滤器类似,共同点都拦截资源,区别:过滤器拦截所有资源,拦截器只拦截action请求,拦截器只在struts中用,而过滤器可以在struts和servlet中用。

用户访问的时候按顺序执行18个拦截器;先执行Action类的创建,在执行拦截器,拦截器执行完后,在执行action类的业务逻辑方法

(2)struts-plugin.xml,     struts相关插件

(3)struts.xml                 用户编写的配置文件

二、访问,可以在HelloAction类中添加一个无参构造方法

public HelloAction() {
// TODO Auto-generated constructor stub
System.out.println("HelloAction.HelloAction()");
}

每当我们访问的时候可以看到控制台的信息,同时不论第一次访问,还是第二次访问,他都会调用这个无参构造方法,与servlet的一次调用不同。

第一个struts案例及分析

所以用户每次访问都会创建一个acton实例。


配置详解

总配置文件,可以包含其他的配置文件

<?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"> <!-- START SNIPPET: xworkSample -->
<struts>
<package name="gqx" extends="struts-default" abstract="false">
<!-- 定义了一个test的action,具体实现com.gqx.action.HelloAction类 -->
<action name="test" class="com.gqx.action.HelloAction" method="execute">
<!-- 配置action返回为success的时候,页面将跳转到success.jsp中 -->
<result name="success">/success.jsp</result>
</action>
</package> <!-- 在总配置文件中引入其他所有的配置文件(src/struts.xml在服务器自动的时候就会执行) -->
<!-- package 定义一个包,包的作用:管理action(通常一个业务模板放一个包) -->
<include file="com/gqx/other/user.xml"></include>
</struts> <!-- END SNIPPET: xworkSample -->

被包含的配置文件(user.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"> <!-- START SNIPPET: xworkSample -->
<struts>
<!-- package 定义一个包,包的作用:管理action(通常一个业务模板放一个包)
name:包的名字;包名不可重复,若重复,服务器启动的时候会报错
extends:当前包继承自哪个包,在struts中一定要继承自struts-default,在struts-default.xml中指定的包,
abstract:表示当前包为抽象包,抽象包中不能有action的定义,否则运行的时候会出错
abstract:true只有当前包被其他包继承的时候使用
namespace:名称空间,是路径的一部分,默认为"/",
如果将默认改为"/user/",则这个action处理类访问的地址就会在http://localhost:8080/项目/名称空间(/user/)/ActionName(login)下访问
action:配置请求路径与action类的映射关系
name:请求的路径名称
class:请求处理的action类的全名
type:跳转的结果类型
标签体中指定跳转的页面 namespace
-->
<package name="user" extends="struts-default" abstract="false">
<action name="login" class="com.gqx.other.UserAction" method="login">
<!-- 配置action返回为success的时候,页面将跳转到success.jsp中 -->
<result name="login">/success.jsp</result>
</action>
</package> </struts> <!-- END SNIPPET: xworkSample -->

这里写了另外一个测试案例,其中包含了对象的封装,还有关于域对象中数据的封装

1、写一个user的bean

package com.gqx.other;

public class User {
private String userName;
private String pwd;
public String getUserName() {
return userName;
}
public void setUserName(String userName) {
this.userName = userName;
}
public String getPwd() {
return pwd;
}
public void setPwd(String pwd) {
this.pwd = pwd;
} }

2、写一个action的类UserAction

package com.gqx.other;

import java.util.Map;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport; //开发action,处理请求
public class UserAction extends ActionSupport{
private User user=new User();
public void setUser(User user) {
this.user = user;
}
public User getUser() {
return user;
} public String login(){
//获取用户的用户名和密码
System.out.println(user.getUserName());
System.out.println(user.getPwd()); //要把数据保存到域中去
ActionContext ac=ActionContext.getContext();
//struts会处理以下域对象
//得到request的map
Map<String, Object> request=ac.getContextMap();
//得到session的map
Map<String, Object> session=ac.getSession();
//得到servletContext的map
Map<String, Object> application=ac.getApplication(); //保存
request.put("rr", "哈哈哈,");
session.put("ss", "Hello ");
application.put("aa", "World!");
return "login";
}
}

配置文件就是上面讲的配置文件,以这个为案例写的,现在去写登入页面,注意登入页面的input中的name

<form action="${pageContext.request.contextPath }/login" name="frmLogin"  method="post">
用户名: <input type="text" name="user.userName"> <br/>
密码: <input type="text" name="user.pwd"> <br/>
<input type="submit" value="登陆"> <br/>
</form>

然后就是index.jsp了

 This is my success page. <br>
${requestScope.rr }
${sessionScope.ss }
${applicationScope.aa }

最后去访问,如图:

第一个struts案例及分析


js小练习

qq显示的效果,这里注意索引的使用,以及this的使用

<!DOCTYPE html>
<html> <head>
<meta charset="utf-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<title>QQ效果</title>
<style type="text/css" media="screen">
div {
display: none;
}
</style>
</head> <body>
<ul>
<li>
<h3>列表1</h3>
<div>
<p>pppppp</p>
<p>pppppp</p>
</div>
</li>
<li>
<h3>列表2</h3>
<div>
<p>pppppp</p>
<p>pppppp</p>
</div>
</li>
<li>
<h3>列表3</h3>
<div>
<p>pppppp</p>
<p>pppppp</p>
</div>
</li>
<li>
<h3>列表4</h3>
<div>
<p>pppppp</p>
<p>pppppp</p>
</div>
</li>
</ul>
<script type="text/javascript">
var h3 = document.getElementsByTagName('h3');
var div = document.getElementsByTagName('div');
for (var i = h3.length - 1; i >= 0; i--) {
h3[i].index = i;
h3[i].onoff = true;
h3[i].onclick = function() {
if (this.onoff) {
div[this.index].style.display = 'inline';
this.onoff = false;
} else {
div[this.index].style.display = 'none';
this.onoff = true;
}
}
}
</script>
</body> </html>