第一个struts案例及分析

Posted 晓乎

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了第一个struts案例及分析相关的知识,希望对你有一定的参考价值。

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

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执行流程

一、服务器启动:

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的一次调用不同。

所以用户每次访问都会创建一个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 }

最后去访问,如图:

 


 

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>

 

以上是关于第一个struts案例及分析的主要内容,如果未能解决你的问题,请参考以下文章

Struts2框架运行流程及案例

Struts2第一个入门案例

Struts2第一个入门案例

struts2第一个入门小案例

Struts2 maven项目简单案例

Struts2 第一个入门小案例