成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Struts2【攔截器】就是這么簡單

chanjarster / 2759人閱讀

摘要:文件上傳通過這個攔截器攔截器的設(shè)計就是基于組件設(shè)計的應(yīng)用再次回顧攔截器基礎(chǔ)在開始講解的時候已經(jīng)說明過了這個文件,它定義了的所有攔截器。由于我們配置了自定義攔截器,那么默認的攔截器棧是不會執(zhí)行的。

什么是攔截器

攔截器Interceptor.....攔截器是Struts的概念,它與過濾器是類似的...可以近似于看作是過濾器

為什么我們要使用攔截器

前面在介紹Struts的時候已經(jīng)講解過了,Struts為我們實現(xiàn)了很多的功能,比如數(shù)據(jù)自動封裝阿..文件上傳功能阿....Struts為我們提供的這些功能都是通過攔截器完成的......

數(shù)據(jù)自動封裝通過這個攔截器。

文件上傳通過這個攔截器

攔截器的設(shè)計就是基于組件設(shè)計的應(yīng)用

再次回顧攔截器基礎(chǔ)

在開始講解Struts的時候已經(jīng)說明過了struts-default.xml這個文件,它定義了Struts的所有攔截器。因為我們在啟動服務(wù)器的時候會自動裝載這個文件,因此我們才可以在Action中使用到Struts為我們提供的功能【數(shù)據(jù)自動封裝...文件上傳】

在struts-default.xml中定義的攔截器就有32個之多,Struts2為了方便我們對攔截器的引用,提供了攔截器棧的定義。

            
                
                
                
                
                
                
                
                
                
                
                
                
                
                
                    dojo..*,^struts..*,^session..*,^request..*,^application..*,^servlet(Request|Response)..*,parameters...*
                
                
                
                    input,back,cancel,browse
                
                
                    input,back,cancel,browse
                
                
            

也就是說:當我們要引用多個攔截器的時候,只要把攔截器都放在棧里頭,在外邊引用攔截器即可!

值得注意的是:Struts2默認執(zhí)行的是默認攔截器棧,一旦用戶有指定執(zhí)行哪些攔截器,那么默認的攔截器棧就不會被執(zhí)行!

自定義攔截器

Struts2允許我們自定義攔截器,這就使我們能夠更加靈活地操作Struts2這個框架了!

Struts2提供了Interceptor這個攔截器接口,只要我們實現(xiàn)這個接口,那么這就算是自定義開發(fā)攔截器了。

當然啦,大部分時候,我們定義攔截器都是繼承AbstractInterceptor這個類....為了學(xué)習(xí)攔截器的內(nèi)容,下面就實現(xiàn)Interceptor這個接口了。

編寫攔截器類

當實現(xiàn)該接口時,有3個需要我們實現(xiàn)的方法:


public class MyInterceptor implements Interceptor {
    @Override
    public void destroy() {

    }

    @Override
    public void init() {

    }

    @Override
    public String intercept(ActionInvocation actionInvocation) throws Exception {
        return null;
    }
}

init()和destory()都是和攔截器執(zhí)行順序有關(guān)的方法,我們現(xiàn)在先不理會....首先來講解intercept這個方法

    /**
     * @param actionInvocation 攔截器的執(zhí)行狀態(tài)
     */
    @Override
    public String intercept(ActionInvocation actionInvocation) throws Exception {
        
        //調(diào)用invoke()方法,代表著放行執(zhí)行下一個攔截器,如果沒有攔截器了,那么就執(zhí)行Action的業(yè)務(wù)代碼
        actionInvocation.invoke();
        return null;
    }

這很容易就能讓我們想起在學(xué)習(xí)過濾器中的doFilter()方法,其實是差不多的!

在struts.xml中配置

像Struts默認的攔截器一樣,我們自定義的攔截器是需要我們在struts中配置的。

由于我們配置了自定義攔截器,那么struts默認的攔截器棧是不會執(zhí)行的。如果我們想要使用默認攔截器棧的功能,就必須把它配置在我們自定義的棧中!

    
        
        
            
            
            
            
            
                
                
                
                
                
            
        
        
        
        
        
        
            /index.jsp

        


    
攔截器的執(zhí)行順序

我們來觀察攔截器和Action類的執(zhí)行順序...只要在對應(yīng)的方法上向控制臺輸出就行了!

攔截器

public class MyInterceptor implements Interceptor {
    @Override
    public void destroy() {

        System.out.println("我是攔截器的銷毀方法");

    }

    @Override
    public void init() {

        System.out.println("我是攔截器的初始化方法");
    }


    /**
     * @param actionInvocation 攔截器的執(zhí)行狀態(tài)
     */
    @Override
    public String intercept(ActionInvocation actionInvocation) throws Exception {

        System.out.println("我是攔截器的攔截方法");

        //調(diào)用invoke()方法,代表著放行執(zhí)行下一個攔截器,如果沒有攔截器了,那么就執(zhí)行Action的業(yè)務(wù)代碼
        //可看成是過濾器的doFilter()方法
        actionInvocation.invoke();
        return null;
    }
}

Action類

public class TestAction extends ActionSupport {

    public TestAction() {
        System.out.println("我是Action類,我被初始化了!");
    }

    @Override
    public String execute() throws Exception {

        System.out.println("我是Action類的執(zhí)行方法");


        return null;

    }
}
效果

從效果圖我們可以看出,他們的執(zhí)行順序是這樣的:

當服務(wù)器開啟的時候,會執(zhí)行攔截器的init()方法

當訪問Action時,Action實例被創(chuàng)建

創(chuàng)建完Action實例,會調(diào)用攔截器的interceptor()方法

最后,執(zhí)行Action的execute()方法

其實很好理解,之前我們使用Struts為我們提供數(shù)據(jù)自動封裝功能的時候,是這樣子的:

服務(wù)器啟動,加載配置文件的信息

初始化默認的攔截器棧

當用戶訪問Action時,創(chuàng)建Action的實例。拿到Action具體的信息【成員變量、setter和getter】

執(zhí)行攔截器具體的內(nèi)容,根據(jù)Action具體的信息,把web端的數(shù)據(jù)封裝到Action上

最后在execute()就可以得到封裝后的數(shù)據(jù)了!

攔截器應(yīng)用案例

需求:當用戶登陸成功,跳轉(zhuǎn)到顯示用戶的JSP頁面中。當用戶登陸失敗,重新返回登陸界面。如果用戶直接訪問顯示用戶的JSP頁面,那么返回到登陸界面

分析

實現(xiàn)這個需求,我們可以使用過濾器的。只要獲取用戶的請求URL,再判斷URL是不是為list.jsp,如果是,我們返回到登陸的界面就好了。

現(xiàn)在,為了對攔截器的理解,我們使用攔截器去完成這個功能!

搭建配置環(huán)境

導(dǎo)入我們c3p0.xml文件

導(dǎo)入c3p0開發(fā)包

導(dǎo)入mysql開發(fā)包

寫數(shù)據(jù)庫連接池工具類

dbUtils開發(fā)包

8個struts2需要用到的開發(fā)包

創(chuàng)建數(shù)據(jù)庫表,導(dǎo)入數(shù)據(jù)

編寫entity
package zhongfucheng.entity;

/**
 * Created by ozc on 2017/5/3.
 */
public class User {
    
    
    private String id ;
    private String username;
    private String cellphone;
    private String email;
    private String password;

    public String getId() {
        return id;
    }

    public void setId(String id) {
        this.id = id;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getCellphone() {
        return cellphone;
    }

    public void setCellphone(String cellphone) {
        this.cellphone = cellphone;
    }

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }

    public String getPassword() {
        return password;
    }

    public void setPassword(String password) {
        this.password = password;
    }
}
編寫DAO
package zhongfucheng.dao;

import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import zhongfucheng.entity.User;
import zhongfucheng.utils.Utils2DB;

import java.sql.SQLException;
import java.util.List;

/**
 * Created by ozc on 2017/5/3.
 */
public class UserDao {

    public User login(User user) {

        try {
            String sql = "SELECT * FROM user WHERE username = ? AND password = ?";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());
            return (User) queryRunner.query(sql, new BeanHandler(User.class), new Object[]{user.getUsername(), user.getPassword()});
        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }

        return null;
    }

    public List getAll() {

        try {
            String sql = "SELECT * FROM user";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());
            return (List) queryRunner.query(sql, new BeanListHandler(User.class));
        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }

        return null;
    }

}
編寫Service
public class Service {

    UserDao userDao = new UserDao();

    public User login(User user) {
        return userDao.login(user);

    }

    public List getAll() {
        return userDao.getAll();
    }
}
編寫登陸的JSP頁面
<%@ page contentType="text/html;charset=UTF-8" language="java" %>


    登陸頁面






編寫處理請求的Action
package zhongfucheng.action;

import com.opensymphony.xwork2.ActionContext;
import zhongfucheng.entity.User;
import zhongfucheng.service.Service;

import java.util.List;
import java.util.Map;

/**
 * Created by ozc on 2017/5/3.
 */
public class UserAction {


    /****************1.封裝數(shù)據(jù)********************/
    private User user;

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    /***************2.調(diào)用Service*******************/
    Service service = new Service();


    //登陸
    public String login() {

        User user = service.login(this.user);

        if (user == null) {
            return "input";
        } else {
            //將user的信息存到Session域?qū)ο笾?            Map session = ActionContext.getContext().getSession();
            session.put("user", user);


            //登陸成功
            return "login";
        }
    }

    //查看user信息
    public String list() {

        //拿到所有用戶的信息
        List users = service.getAll();

        //存到request域?qū)ο笾?        Map request = ActionContext.getContext().getContextMap();

        request.put("users", users);

        return "list";
    }

}
struts.xml配置文件
    
        

            
            user_list

            
            /WEB-INF/list.jsp
        

    

到目前為止,我們登陸或者不登陸都可以得到用戶的具體信息....這是不合理的

我們想要的效果是:只有用戶正在調(diào)用login方法,或者該用戶已經(jīng)登陸了,才可以查看具體的用戶信息。

因此,我們們要攔截它們,只有用戶調(diào)用的是login方法時或者已經(jīng)登陸的情況下,才能跳轉(zhuǎn)到對應(yīng)的顯示頁面

攔截器
package zhongfucheng;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.ActionProxy;
import com.opensymphony.xwork2.interceptor.AbstractInterceptor;

/**
 * Created by ozc on 2017/5/3.
 */
public class Interceptor  extends AbstractInterceptor{


    @Override
    public String intercept(ActionInvocation actionInvocation) throws Exception {

        //得到正在執(zhí)行的代理對象
        ActionProxy proxy = actionInvocation.getProxy();

        //通過代理對象得到正在執(zhí)行的方法
        String method = proxy.getMethod();


        //如果方法的名字不是login,那么就讓他們返回到login頁面上
        if (!method.equals("login")) {

            //查看用戶是否登陸了
            Object user = ActionContext.getContext().getSession().get("user");

            //如果沒有登陸,回到login頁面
            if (user == null) {

                return "input";
            } else {

                //登陸了,那么就讓它訪問具體的用戶信息頁面
                return actionInvocation.invoke();
            }
        } else {

            //如果是訪問login方法,那么就讓它執(zhí)行
            return actionInvocation.invoke();
        }

    }
}
Struts.xml




    

        
            
            

            
            
                
                
            

        


        

        

        
        

        

            
            user_list

            
            /WEB-INF/list.jsp


            
            /login.jsp

        


    

效果:

只有當用戶登陸了才能查看用戶具體信息,直接訪問Action會跳轉(zhuǎn)回

Struts2其他攔截器 計時攔截器

Struts2自帶了計時攔截器,也就是用來統(tǒng)計每個Action執(zhí)行的時間

執(zhí)行等待攔截器

如果頁面執(zhí)行得太慢了,Struts2還提供了執(zhí)行等待攔截器,也就是說,當頁面加載得太久了,就跳轉(zhuǎn)到對應(yīng)的提示頁面...當服務(wù)器執(zhí)行完畢了,也跳轉(zhuǎn)到相對應(yīng)的頁面

Struts2防止表單重復(fù)提交攔截器 回顧防止表單重復(fù)提交

當我們學(xué)習(xí)Session的時候已經(jīng)通過Session來編寫了一個防止表單重復(fù)提交的小程序了,我們來回顧一下我們當時是怎么做的:

在Servlet上生成獨一無二的token,保存在Session域中,并交給JSP頁面

JSP頁面在提交表單數(shù)據(jù)的時候,把token放在隱藏域中...一起帶過去給Servlet

Servlet判斷用戶有沒有帶token值過來,判斷token的值是否和Session的相匹配

如果用戶是第一次提交的話,那么就允許用戶的請求,接著就把保存在Session中的token值去除

等用戶想要再次提交的時候,Servlet發(fā)現(xiàn)Session中并沒有token了,所以不搭理用戶的請求

我們以前寫表達重復(fù)提交就花了這么幾個步驟...如果有興趣的同學(xué)可以看一下以前的實現(xiàn)思路:http://blog.csdn.net/hon_3y/article/details/54799494#t11

Struts2防止表單重復(fù)提交

Struts2是簡化我們的開發(fā)的,表單重復(fù)提交也是一件非常常用的功能...Struts2也為我們實現(xiàn)了...當然啦,也是通過攔截器來實現(xiàn)

   

它的實現(xiàn)原理和我們以前寫的思路幾乎一致...它不需要另外寫一個組件來生成token值,struts2標簽就有這么一個功能...因此是十分方便的

為了熟悉一下Struts2,我們也使用Struts2來編寫一下上圖的程序...

編寫DAO
package zhongfucheng.dao;

import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.handlers.BeanHandler;
import org.apache.commons.dbutils.handlers.BeanListHandler;
import zhongfucheng.entity.User;
import zhongfucheng.utils.Utils2DB;

import java.sql.SQLException;
import java.util.List;

/**
 * Created by ozc on 2017/5/3.
 */
public class UserDao {

    public void add(User user) {
        try {

            String sql = "INSERT INTO user(id,username,cellphone,password,address) VALUES (?,?,?,?,?)";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());

            queryRunner.update(sql, new Object[]{user.getId(), user.getUsername(), user.getCellphone(), user.getPassword(),user.getAddress()});

        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }
    }

    public User findUser(String id) {
        try {
            String sql = "SELECT * FROM user WHERE id=?";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());

            return (User) queryRunner.query(sql, new BeanHandler(User.class), new Object[]{id});
            
        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }
        return null;
    }

    public List getAll() {

        try {
            String sql = "SELECT * FROM user";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());
            return (List) queryRunner.query(sql, new BeanListHandler(User.class));
        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }
        return null;
    }
    public void  updateUser(User user) {

        try {
            String sql = "UPDATE user SET username=?,password=?,cellphone=? WHERE id=?";
            QueryRunner queryRunner = new QueryRunner(Utils2DB.getDataSource());

            queryRunner.update(sql, new Object[]{user.getUsername(), user.getPassword(), user.getCellphone(), user.getId()});
        } catch (SQLException e) {
            new RuntimeException("登陸失敗了!");
        }
    }

}
編寫service
package zhongfucheng.service;

import zhongfucheng.dao.UserDao;
import zhongfucheng.entity.User;
import zhongfucheng.utils.WebUtils;

import java.util.List;

/**
 * Created by ozc on 2017/5/3.
 */
public class Service {

    UserDao userDao = new UserDao();

    public void add(User user) {


        //手動設(shè)置id,因為在數(shù)據(jù)庫表我沒使用自動增長id
        user.setId(WebUtils.makeId());

        //這是以前的表,規(guī)定要address,只能手動設(shè)置了
        user.setAddress("廣州");
        userDao.add(user);

    }

    public User findUser(String id) {

        return userDao.findUser(id);

    }

    public List getAll() {

        return userDao.getAll();

    }
    public void  updateUser(User user) {


        userDao.updateUser(user);

    }
}
開發(fā)步驟

編寫添加用戶JSP

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@taglib prefix="s" uri="/struts-tags" %>





用戶名:
密碼:
電話:

使用了模型驅(qū)動封裝數(shù)據(jù),添加用戶


    //這里一定要實例化
    User user = new User();

    public User getUser() {
        return user;
    }

    public void setUser(User user) {
        this.user = user;
    }

    @Override
    public User getModel() {
        return user;
    }


    /*******調(diào)用service********/
    Service service = new Service();

    public String register() throws Exception {

        service.add(user);


        //注冊成功,就跳轉(zhuǎn)到list()方法,list方法就跳轉(zhuǎn)到查看所有用戶頁面了!
        return list();
    }

列出全部的用戶數(shù)據(jù),提供修改功能,需要把id傳遞過去,明確修改的是哪一個用戶

<%--
  Created by IntelliJ IDEA.
  User: ozc
  Date: 2017/5/2
  Time: 18:24
  To change this template use File | Settings | File Templates.
--%>
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
<%@taglib prefix="fn" uri="http://java.sun.com/jsp/jstl/functions" %>
<%@ taglib prefix="s" uri="/struts-tags" %>


    列出下載頁面



        
            
用戶id 用戶姓名 用戶密碼 用戶電話 操作
${user.id} ${user.username} ${user.password} ${user.cellphone} 修改

Action得到web帶過來的id,找到對象,添加到值棧中(數(shù)據(jù)回顯)

    public String updatePage() throws Exception {

        //得到用戶帶過來的id,根據(jù)id查找對象
       User user222 = service.findUser(user.getId());

        ActionContext.getContext().getValueStack().push(user222);

        return "updatePage";
    }

修改用戶的JSP頁面,使用Struts2提供的回顯技術(shù),并把id通過隱藏域帶過去給Action..最終是通過id來修改用戶的數(shù)據(jù)

用戶名
密碼
電話
效果

防止表單重復(fù)提交

上面我們已經(jīng)完成了大部分的功能了,但當我們如果提交之后,再刷新頁面,那么表單的數(shù)據(jù)就會重復(fù)提交...我們使用Struts2我們提供的防止表單重復(fù)提交的功能把!

在需要提交的表單上使用token標簽
用戶名:
密碼:
電話:
在struts配置文件中配置攔截器

token攔截器默認是不會啟動的,也就是說:需要我們手動配置...

當我們配置攔截器的時候,Struts2默認的攔截器是不會執(zhí)行的,所以要把Struts2默認的攔截器也寫上





    
    



        

            

            
                
                register
            

            
             /list.jsp

            
            /update.jsp

            
            /login.jsp

            /user_list
        
    

    


當我們重復(fù)提交的時候,它會報錯,因此,如果它報錯了,我們就跳轉(zhuǎn)到register頁面把

測試

如果文章有錯的地方歡迎指正,大家互相交流。習(xí)慣在微信看技術(shù)文章,想要獲取更多的Java資源的同學(xué),可以關(guān)注微信公眾號:Java3y

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/68742.html

相關(guān)文章

  • Java3y文章目錄導(dǎo)航

    摘要:前言由于寫的文章已經(jīng)是有點多了,為了自己和大家的檢索方便,于是我就做了這么一個博客導(dǎo)航。 前言 由于寫的文章已經(jīng)是有點多了,為了自己和大家的檢索方便,于是我就做了這么一個博客導(dǎo)航。 由于更新比較頻繁,因此隔一段時間才會更新目錄導(dǎo)航哦~想要獲取最新原創(chuàng)的技術(shù)文章歡迎關(guān)注我的公眾號:Java3y Java3y文章目錄導(dǎo)航 Java基礎(chǔ) 泛型就這么簡單 注解就這么簡單 Druid數(shù)據(jù)庫連接池...

    KevinYan 評論0 收藏0
  • SpringMVC入門就這么簡單

    摘要:也就是說映射器就是用于處理什么樣的請求提交給處理。這和是一樣的提交參數(shù)的用戶名編號提交配置處理請求注冊映射器包框架接收參數(shù)設(shè)置無參構(gòu)造器,里邊調(diào)用方法,傳入要封裝的對象這里的對象就表示已經(jīng)封裝好的了對象了。 什么是SpringMVC? SpringMVC是Spring家族的一員,Spring是將現(xiàn)在開發(fā)中流行的組件進行組合而成的一個框架!它用在基于MVC的表現(xiàn)層開發(fā),類似于struts...

    SKYZACK 評論0 收藏0
  • 納稅服務(wù)系統(tǒng)【總結(jié)】

    摘要:要是使用到日歷的話,我們想到使用這個日歷類上面僅僅是我個人總結(jié)的要點,如果有錯誤的地方還請大家給我指正。 納稅服務(wù)系統(tǒng)總結(jié) 納稅服務(wù)系統(tǒng)是我第一個做得比較大的項目(不同于javaWeb小項目),該項目系統(tǒng)來源于傳智Java32期,十天的視頻課程(想要視頻的同學(xué)關(guān)注我的公眾號就可以直接獲取了) 我跟著練習(xí)一步一步完成需求,才發(fā)覺原來Java是這樣用來做網(wǎng)站的,Java有那么多的類庫,頁面...

    ispring 評論0 收藏0
  • SpringMVC【開發(fā)Controller】詳解

    摘要:是使用攔截器來自動幫我們完成中文亂碼的問題的。這是我的首頁當然了,基于注解和基于來開發(fā),都是通過映射器適配器和視圖解析器的。能夠控制請求路徑和請求方式一個控制器寫多個業(yè)務(wù)方法到目前為止,我們都是一個控制器寫一個業(yè)務(wù)方法,這肯定是不合理的。 前言 本文主要是講解在Controller中的開發(fā),主要的知識點有如下: 編碼過濾器 使用注解開發(fā) 注解@RequestMapping詳解 業(yè)務(wù)方...

    Pines_Cheng 評論0 收藏0
  • Struts2截器

    摘要:自定義的攔截器可以和框架內(nèi)置的攔截器進行混合使用,一般情況攔截器都被默認配置成為執(zhí)行的基礎(chǔ)。若類型轉(zhuǎn)換失敗,或者數(shù)據(jù)驗證失敗,攔截器就會阻止的執(zhí)行。 1.攔截器簡介 默認的攔截器在設(shè)計的時候就能滿足大部分的應(yīng)用,所以很多時候就不需要添加自定義的攔截器或者修改攔截器棧。很多action有各種各樣的需求,比如輸入驗證、文件上傳、防止多次提交等等。于是struts框架就提供了一個解決方案,I...

    dendoink 評論0 收藏0

發(fā)表評論

0條評論

chanjarster

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<