struts2入门学习

本篇博客知识点
1.struts是什么?
2.Struts2框架主要由三部分组成
3.Struts2框架的处理流程
4.Struts2入门示例
5.Struts2其他功能
struts是什么?

Struts2是Struts的第二代产品,以WebWork为核心,采用拦截器的机制处理用户请求,使业务逻辑控制器能与Servlet API完全脱离。Struts1采用Servlet的机制处理用户请求。
     Struts 2框架的所有类都基于接口,核心接口独立于HTTP。Struts 2配置文件中的大多数配置元素都会有默认值,有助于减少在XML文件中需要进行的配置。
     简单来说 就是 把JavaWeb中的MVC框架中Servlet和JavaBean整合到一起成为一个action,同时struts2还帮我们把前端的数据已经封装好了,这一切都是通过xml形式的配置文件完成的

Struts2框架主要由三部分组成:

核心控制器(StrutsPrepareAndExecuteFilter)、业务控制器和用户定义的业务逻辑组件。
    (也有核心控制器使用FilterDispatcher)

核心控制器

FilterDispatcher是早期struts2的过滤器,可以对客户端URL请求进行过滤,负责处理用户所有以.action结尾的请求。2.1.3版后,官方推荐使用StrutsPrepareAndExecuteFilter

业务控制器

是用户实现的Action类实例。Action类通常包含一个execute方法,返回一个字符串作为逻辑视图名。创建了Action类之后,还需要在struts.xml文件中配置此Action的相关信息

业务逻辑组件

通常是指用户自己针对系统功能开发的功能模块组件。被业务控制器组件所调用来处理业务逻辑的。

整个的逻辑流程图如下
这里写图片描述

Struts2框架的处理流程

1步:客户端浏览器发送一个请求。 

第2步: web服务器如Tomcat收到该请求,读取配置文件,将请求 导向Struts2的StrutsPrepareAndExecuteFilter(核心控制器), 后者根据请求决定调用合适Action。 

第3步:StrutsPrepareAndExecuteFilter在调用Action之前被Struts2的拦截器拦截,拦截器自动对请求应用通用功能,如数据转换,校验等。

第4步:调用Action的execute方法,该方法根据请求的参数来执行一定的操作。

第5步:依据Action的execute方法处理结果,导向不同的URL。如在execute中验证用户,验证成功可以导向成功的页面。否则重新登录

Struts2入门示例—用来体会struts2的工作思想

用struts2实现用户登录并导向不同结果页面

第一步:你需要准备一个官方下载的struts2的空项目区官网下载就好了
这里写图片描述

第二步:点开解压,struts-2.3.20->apps->struts2-blank.war找到一个空的模板项目解压
这里写图片描述
第三步:可以新建自己web项目,所需要的包就在第二步的空项目的lib目录里面,其配置文件struts.xml也是。
这里写图片描述

这里写图片描述

第四步:把web.xml用struts的过滤器拦了,

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="3.0" 
    xmlns="http://java.sun.com/xml/ns/javaee" 
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
    http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd">
  <display-name></display-name> 

    <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.jsp</welcome-file>
  </welcome-file-list>
</web-app>

第五步:写action

package cn.hncu.login.action;

import java.util.Map;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;


//EJB  POJO(属性名和页面提交参数的名字相同,必须要写getter-setter方法)+execute()

//如果Action中的所有功能都是自己写,那么不需要继承任何类。
//但是,如果继承了ActionSupport类,则可以获得该父带来的一些功能,如数据校验等
public class LoginAction extends ActionSupport{
    private String username;
    private String password;

    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }

    public String execute(){//根据默认业务方法返回的字符串来指定结果页面
        //System.out.println(username+","+password);

        //我想自己定义一些数据放到指定的容器中
        ActionContext context = ActionContext.getContext();
        //往app容器中放
        Map<String,Object>  appCtx = context.getApplication();
        appCtx.put("user", this);  //类似servlet中的app.setAttribute("user",this);
        //往session窗口中放
        context.getSession().put("user", this);
        context.getSession().put("aaa", "James");
        context.getApplication().put("NBA", "Kobe");

        //手动获取指定容器中的数据
        Object obj = context.getSession().get("user");



        System.out.println("访问service层....这里略过....");
        if(username.startsWith("hncu") && password.length()>3){
            return "success";
        }else{
            return "index";
        }

    }

    @Override
    public void validate() {
        System.out.println("数据校验:"+username+"---"+password);
    }

    public void fun1(){
        System.out.println("fun1:"+username+","+password);
    }
}

第六步:写struts.xml 这个的作用就是控制中心,src目录下

<?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>
    <!-- 如果想访问action中指定的业务方法,而不是默认的execute方法,
            则要打开下面这个开关。如果业务方法名想用通配符的方式访问则又要关闭该配置项! -->
    <constant name="struts.enable.DynamicMethodInvocation" value="true" />

    <package name="default" namespace="/" extends="struts-default">
        <action name="login" class="cn.hncu.login.action.LoginAction">
            <!-- 大小写必须和execute()方法的返回值完全一样  -->
            <result  name="success">/jsps/show.jsp</result>
            <result  name="index">/index.jsp</result>
        </action>        
          </package>   
</struts>

第七步:把结果页面写了 show.jsp 还有index.jsp
action后面的请求的就是前面配置的action的名字

<form action="login">
                用户名:<input type="text" name="username"/> <br/>
                密码:<input type="password" name="password"/> <br/>
                <input type="submit" value="提交">
            </form>
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>

<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
  </head>

  <body>
    <h2>恭喜,登录成功!!!</h2>
       欢迎你:${username}  <br/>
    ${sessionScope.name},,,${NBA} ..${user.username }

  </body>
</html>

大致流程如上,总的来说不难,但是会有很多细节,配置文件有一错误就会404.。。需要多次练习慢慢来,保持耐心吧。 都是坑过来的。

下面我想记录一下struts2的其他功能

sttuts2的校验器功能
如原本的action如下

package com.action;
import com.opensymphony.xwork2.ActionSupport;
public class LoginValidateAction extends ActionSupport{
    // Action类公用私有变量,用来做页面导航标志
    private static String FORWARD = null;
    private String username;
    private String password;
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String getPassword() {
        return password;
    }
    public void setPassword(String password) {
        this.password = password;
    }
    public void validate() {

    }

    public String execute() throws Exception {
        username = getUsername();   //属性值即JSP页面上输入的值
        password = getPassword();       //属性值即JSP页面上输入的值
        try {
            // 判断输入值是否是空对象或没有输入
            if (username.equals("admin")&& password.equals("1234")) {
            // 根据标志内容导航到操作成功页面
                FORWARD = "success";
            } else {
            // 根据标志内容导航到操作失败页面
                FORWARD = "input";
            }
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        return FORWARD;
    }
}

配置一个校验器的xml文档:action名+“-validation.xml”

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE validators PUBLIC
        "-//OpenSymphony Group//XWork Validator 1.0.3//EN"
        "http://www.opensymphony.com/xwork/xwork-validator-1.0.3.dtd">
<validators>
    <field name="username">
        <field-validator type="requiredstring">
            <message key="用户名不能为空"/>
        </field-validator>
    </field>
    <field name="password">
        <field-validator type="requiredstring">
            <message key="密码不能为空"/>
        </field-validator>
        <field-validator type="stringlength">
            <param name="minLength">6</param>
            <param name="maxLength">16</param>
            <message>密码长度应在6~16个字符之间</message>
        </field-validator>
    </field>
</validators>

页面的 话就需要用淡他标签库里的东西

<%@ page language="java" pageEncoding="utf-8"%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<!-- struts2标签库调用声明 -->
<%@taglib prefix="s" uri="/struts-tags"%>
<html>
<head>
    <title>登录页面</title>
</head>
<body>
    <!-- form标签库定义,以及调用哪个Action声明 -->
    <s:form action="validate">
        <table width="60%" height="76" border="0">
                <!-- 各标签定义 -->
                <s:textfield name="username" label="用户名"/>
                <s:password name="password" label="密  码" />
                <s:submit value="登录" align="center"/>               
        </table>
    </s:form>
</body>
</html>

当然里,struts文件里action也是要加的

</action>
          <action name="validate" class="类全名">
            <result name="input">/loginvalidate.jsp</result>
            <result>/index.jsp</result>
         </action>

文件上传

package com.action;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.OutputStream;

import org.apache.struts2.ServletActionContext;

import com.opensymphony.xwork2.ActionSupport;

public class FileUploadAction extends ActionSupport {
    private static final int BUFFER_SIZE = 16 * 1024;
    private String title; // 文件标题
    private File upload; 
    private String uploadFileName; // 上传文件名=  upload+"FileName"
    private String uploadContentType; // 上传文件类型=  upload+"ContentType"

    private String savePath; // 保存文件的目录路径(通过依赖注入)


    private static void copy(File src, File dst) {
        InputStream in = null;
        OutputStream out = null;
        if(!dst.exists()){
            dst.getParentFile().mkdirs();
        }
        try {
            in = new BufferedInputStream(new FileInputStream(src), BUFFER_SIZE);
            out = new BufferedOutputStream(new FileOutputStream(dst),
                    BUFFER_SIZE);
            byte[] buffer = new byte[BUFFER_SIZE];
            int len = 0;
            while ((len = in.read(buffer)) > 0) {
                out.write(buffer, 0, len);
            }
            in.close();
            out.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public String execute() throws Exception {
        // 根据服务器的文件保存地址和原文件名创建目录文件全路径
        String dstPath = ServletActionContext.getServletContext().getRealPath(
                this.getSavePath())
                + "/" + this.getUploadFileName();

        System.out.println("上传的文件的类型:" + this.getUploadContentType());
        File dstFile = new File(dstPath);
        copy(this.upload, dstFile);
        return SUCCESS;
    }

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public File getUpload() {
        return upload;
    }

    public void setUpload(File upload) {
        this.upload = upload;
    }

    public String getUploadFileName() {
        return uploadFileName;
    }

    public void setUploadFileName(String uploadFileName) {
        this.uploadFileName = uploadFileName;
    }

    public String getUploadContentType() {
        return uploadContentType;
    }

    public void setUploadContentType(String uploadContentType) {
        this.uploadContentType = uploadContentType;
    }

    public String getSavePath() {
        return savePath;
    }

    public void setSavePath(String savePath) {
        this.savePath = savePath;
    }
}

struts配置一下

<action name ="fileUpload" class ="类全名"> 
            <!-- 动态设置Action中的savePath属性的值 --> 
            <param name="savePath">/upload</param> 
            <result name ="success">/index.jsp</result> 
         </action> 

页面上传代码

<%@ page language="java" contentType="text/html; charset=UTF-8"%> 
<html> 
<head> 
    <title>Struts2 File Upload</title> 
</head> 
<body> 
    <form action="fileUpload " method="POST" enctype="multipart/form-data"> 
        文件标题:<input type="text" name="title" size="50"/><br/> 
        选择文件:<input type="file" name="upload" size="50"/><br/> 
       <input type="submit" value=" 上传 "/>       
    </form> 
</body> 
</html> 

拦截器功能
正常需要执行的action

package com.action;
import com.opensymphony.xwork2.ActionSupport;
public class  InterceptorTest extends ActionSupport {
    private String username;
    //private MyDate birth;//特殊类型需要类型转换器
    public String getUsername() {
        return username;
    }
    public void setUsername(String username) {
        this.username = username;
    }
    public String execute() throws Exception {
         System.out.println("此时所有拦截器完毕,调用action中的execute方法");
         return SUCCESS;
    }

}

拦截器代码

package com.interceptor;

import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.Interceptor;

public class MyInterceptor1 implements Interceptor {
    public void init() {// 覆盖Interceptor接口中的init函数
        System.out.println("拦截器已经被加载");
    }

    public void destroy() {// 覆盖Interceptor接口中的destroy函数
        System.out.println("destroy");
    }

    /* 覆盖Interceptor接口中的intercept函数 */
    public String intercept(ActionInvocation invocation) throws Exception {
        System.out.println("调用intercept方法");
        /* invocation.invoke()方法检查是否还有拦截器 有的话继续调用余下的拦截器 没有了则执行action的业务逻辑 */
        String result = invocation.invoke();//放行
        System.out.println("2222222");
        return result;
    }
}

struts里配置拦截器

</package>
     <package name="myinterceptor" extends="struts-default">
        <!-- 定义拦截器 -->
        <interceptors>
          <interceptor name="myInterceptor" class="com.interceptor.MyInterceptor1"/>
        </interceptors>
        <!-- 配置action -->
        <action name="test_interceptor" class="com.action.InterceptorTest">
            <result name="success">/interceptorsuccess.jsp</result>
            <result name="input">/test_interceptor.jsp</result>
            <!-- 将声明好的拦截器插入action中 -->
            <interceptor-ref name="myInterceptor" />
            <interceptor-ref name="defaultStack" />
        </action>
    </package>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值