vue+elementUI完成註冊及登陸

淼呆發表於2019-08-11

vue怎麼引入和配置使用element-ui框架

  1.1 使用vue-cli腳手架工具建立一個vue專案
      vue init webpack pro01

  1.2 npm安裝elementUI
      cd pro01                                   #進入新建專案的根目錄
      npm install element-ui -S                  #安裝element-ui模組

  ## 重要的事情說三遍:在指定位置!!!在指定位置!!!在指定位置!!!~~~新增三行程式碼
  1.3 在專案中src目錄下找到main.js,並在指定位置新增三行程式碼(main.js是入口檔案,所以在這裡引入就行,頁面就不用引入了)
      import Vue from 'vue'
 
      import ElementUI from 'element-ui' //新新增1
      import 'element-ui/lib/theme-chalk/index.css' //新新增2,避免後期打包樣式不同,要放在import App from './App';之前
      
      import App from './App'
      import router from './router'


      Vue.use(ElementUI)   //新新增3
      Vue.config.productionTip = false

  1.4 測試
      修改HelloWorld.vue新增elementUI元件檢視效果

Vue+ElementUI設計登陸頁面

注1:相關樣式見資料“css.txt”
注2:<style scoped>
        在vue元件中,在style標籤上新增scoped屬性,以表示它的樣式作用於當下的模組,很好的實現了樣式私有化的目的
注2:auto-complete="off"
        autocomplete 屬性是 HTML5 中的新屬性,off-----禁用自動完成
http://localhost:8080/T226_easyui/userAction.action?methodName=login&&uid=001&&upwd=123456

後臺互動(axios/qs/vue-axios)

      axios是vue2提倡使用的輕量版的ajax。它是基於promise的HTTP庫。它會從瀏覽器中建立XMLHttpRequests,與Vue配合使用非常好。
   
      1.題外話:
      vue.js有著名的全家桶系列:vue-router,vuex, vue-resource,再加上構建工具vue-cli,就是一個完整的vue專案的核心構成。
      其中vue-resource是Vue.js的一款外掛,它可以通過XMLHttpRequest或JSONP發起請求並處理響應,但在vue更新到2.0之後,
      作者就宣告不再對vue-resource更新,而是推薦的axios

      2.GET提交
      axios.get('/user', {//注意資料是儲存到json物件的params屬性
        params: {
          ID: 12345
        }
      }).then(function (response) {
        console.log(response);
      }).catch(function (error) {
        console.log(error);
      });

      3.POST提交
      axios.post('/user', {//注意資料是直接儲存到json物件
        firstName: 'Fred',
        lastName: 'Flintstone'
      }).then(function (response) {
        console.log(response);
      }).catch(function (error) {
        console.log(error);
      });

      注1:axios跨域問題
           會一直報錯:“http://127.0.0.1:8848' has been blocked by CORS policy: No 'Access-Control-Allow-Origin' header”
           因為使用了前後端分離開發,跨域訪問了,解決方案:需要配置tomcat允許跨域訪問,
           tomcat跨域配置方法很多,但最簡單的方式是自己寫一個過濾器CorsFilter實現,新增一個響應頭
           Access-Control-Allow-Origin即可
           httpResponse.addHeader("Access-Control-Allow-Origin", "*");//*表示任何域名
           httpResponse.addHeader("Access-Control-Allow-Origin", "http://localhost:80"); 

           Access-Control-Allow-Origin:*                           #則允許所有域名的指令碼訪問該資源。
           Access-Control-Allow-Origin:https://www.fujieace.com    #允許特定的域名訪問 

      注2:axios.get提交沒有問題,axios.post提交後臺接收不到資料
           因為POST提交的引數的格式是Request Payload,這樣後臺取不到資料的(詳情見資料“05 Vue中axios踩坑之路-POST傳參 - RainSun - CSDN部落格.mht”),
           解決方案:使用qs.js庫,將{a:'b',c:'d'}轉換成'a=b&c=d'

      注3:為簡化axios使用,還可以使用axios全域性配置及攔截器,詳情見資料“api/http.js”
           axios.defaults.baseURL = 'https://api.example.com';
           //axios.defaults.headers.common['Authorization'] = AUTH_TOKEN;//自定義請求頭,新增認證令牌
           axios.defaults.headers.post['Content-Type'] = 'application/x-www-form-urlencoded';

      注4:為方便開發,建議對請求地址進行封裝,詳情見資料“api/action.js”

      注5:^_^~~~~為進一步簡化開發,將action.js的URL地址封裝到axios.urls屬性上         
     
  3.2 qs
      qs.js它是一個url引數轉化的js庫。用法就兩個:
      var obj = qs.parse('a=b&c=d');  //將URL解析成物件的形式:{a:'b',c:'d'}
      var str = qs.stringify(obj);    //將物件 序列化成URL的形式,以&進行拼接:a=b&c=d'

  3.3 vue-axios
      vue-axios是在axios基礎上擴充套件的外掛,在Vue.prototype原型上擴充套件了$http等屬性,可以更加方便的使用axios

  3.4 axios/qs/vue-axios安裝及使用步驟
      1.安裝
      npm install axios -S
      npm install qs -S  
      npm install vue-axios -S                        
   
      2.修改main.js檔案
      #import axios from 'axios'
      #import qs from 'qs'
      import axios from '@/api/http'             #vue專案對axios的全域性配置      
      import VueAxios from 'vue-axios' 
      Vue.use(VueAxios,axios)

      3.之後就可以直接在Vue元件檔案的methods裡使用了  
      this.axios.get(url,{params:{id:1,name:'zs'}}).then((response)=>{
        this.newsList=response.data.data;
      }).catch((response)=>{
        console.log(response);
      }); 

資料 css.txt

App.vue

<style> 
html,
body {
    width: 100%;
    height: 100%;
    box-sizing: border-box;
    padding: 0px;
    margin: 0px;
}
#app {
    font-family: "Avenir", Helvetica, Arial, sans-serif;
    -webkit-font-smoothing: antialiased;
    -moz-osx-font-smoothing: grayscale;
    color: #2c3e50;
    widows: 100%;
    height: 100%;
}
</style>


login.vue
<style scoped>
	.login-wrap {
		box-sizing: border-box;
		width: 100%;
		height: 100%;
		padding-top: 10%;
		background-image: url();
		/* background-color: #112346; */
		background-repeat: no-repeat;
		background-position: center right;
		background-size: 100%;
	}

	.login-container {
		border-radius: 10px;
		margin: 0px auto;
		width: 350px;
		padding: 30px 35px 15px 35px;
		background: #fff;
		border: 1px solid #eaeaea;
		text-align: left;
		box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
	}

	.title {
		margin: 0px auto 40px auto;
		text-align: center;
		color: #505458;
	}
</style>

下面展示一下前端程式碼

結構如下
在這裡插入圖片描述

action.js

/**
 * 對後臺請求的地址的封裝,URL格式如下:
 * 模組名_實體名_操作
 */
export default {
	'SERVER': 'http://localhost:8080/TT_Ssh', //伺服器
	'SYSTEM_USER_DOLOGIN': '/vue/userAction_login.action', //使用者登陸
	'SYSTEM_USER_DOREG': '/vue/userAction_reg.action', //使用者註冊
	'SYSTEM_MENU_TREE': '/vue/treeNodeAction.action', //左側樹形選單載入
	'SYSTEM_ARTICLE_LIST': '/vue/articleAction_list.action', //文章列表
	'SYSTEM_ARTICLE_ADD': '/vue/articleAction_add.action', //文章新增
	'SYSTEM_ARTICLE_EDIT': '/vue/articleAction_edit.action', //文章修改
	'SYSTEM_ARTICLE_DEL': '/vue/articleAction_del.action', //文章刪除
	'SYSTEM_USER_GETASYNCDATA': '/vue/userAction_getAsyncData.action', //vuex中的非同步載入資料
	'getFullPath': k => { //獲得請求的完整地址,用於mockjs測試時使用
		return this.SERVER + this[k];
	}
}

http.js

/**
 * vue專案對axios的全域性配置
 */
import axios from 'axios'
import qs from 'qs'

//引入action模組,並新增至axios的類屬性urls上
import action from '@/api/action'
axios.urls = action

// axios預設配置
axios.defaults.timeout = 10000; // 超時時間
// axios.defaults.baseURL = 'http://localhost:8080/j2ee15'; // 預設地址
axios.defaults.baseURL = action.SERVER;

//整理資料
// 只適用於 POST,PUT,PATCH,transformRequest` 允許在向伺服器傳送前,修改請求資料
axios.defaults.transformRequest = function(data) {
	data = qs.stringify(data);
	return data;
};


// 請求攔截器
axios.interceptors.request.use(function(config) {
	// var jwt = window.vm.$store.getters.getJwt;
	// config.headers['jwt'] = jwt;
	return config;
}, function(error) {
	return Promise.reject(error);
});

// 響應攔截器
axios.interceptors.response.use(function(response) {
	// debugger;
	// var jwt = response.headers['jwt'];
	// if(jwt){
	// 	window.vm.$store.commit('setJwt',{jwt:jwt});
	// }
	return response;
}, function(error) {
	return Promise.reject(error);
});

// // 路由請求攔截
// // http request 攔截器
// axios.interceptors.request.use(
// 	config => {
// 		//config.data = JSON.stringify(config.data);  
// 		//config.headers['Content-Type'] = 'application/json;charset=UTF-8';
// 		//config.headers['Token'] = 'abcxyz';
// 		//判斷是否存在ticket,如果存在的話,則每個http header都加上ticket
// 		// if (cookie.get("token")) {
// 		// 	//使用者每次操作,都將cookie設定成2小時
// 		// 	cookie.set("token", cookie.get("token"), 1 / 12)
// 		// 	cookie.set("name", cookie.get("name"), 1 / 12)
// 		// 	config.headers.token = cookie.get("token");
// 		// 	config.headers.name = cookie.get("name");
// 		// }
// 		return config;
// 	},
// 	error => {
// 		return Promise.reject(error.response);
// 	});

// // 路由響應攔截
// // http response 攔截器
// axios.interceptors.response.use(
// 	response => {
// 		if (response.data.resultCode == "404") {
// 			console.log("response.data.resultCode是404")
// 			// 返回 錯誤程式碼-1 清除ticket資訊並跳轉到登入頁面
// 			//      cookie.del("ticket")
// 			//      window.location.href='http://login.com'
// 			return
// 		} else {
// 			return response;
// 		}
// 	},
// 	error => {
// 		return Promise.reject(error.response) // 返回介面返回的錯誤資訊
// 	});



export default axios;

index.js

import Vue from 'vue'
import Router from 'vue-router'
import HelloWorld from '@/components/HelloWorld'
import Login from '@/views/Login'
import Reg from '@/views/Reg'


Vue.use(Router)

export default new Router({
	routes: [{
		path: '/',
		name: 'Login',
		component: Login
	},{
		path: '/Login',
		name: 'Login',
		component: Login
	},
	{
		path: '/Reg',
		name: 'Reg',
		component: Reg
	}]
})

Login.vue

<template>
	<div class="login-wrap">
		<el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container">
			<h3 style="text-align: center;">使用者登入</h3>
			<el-form-item label="使用者名稱">
				<el-input v-model="ruleForm.uname"></el-input>
			</el-form-item>
			<el-form-item label="密碼">
				<el-input type="password" v-model="ruleForm.pwd"></el-input>
			</el-form-item>
			<el-form-item>
				<el-row>
					<el-col :span="24">
						<div class="grid-content bg-purple-dark">
							<el-button type="primary" style="width:100% ;" @click="doLogin">提交</el-button>
						</div>
					</el-col>
				</el-row>
				<el-row>
					<el-col :span="12">
						<div class="grid-content bg-purple-dark">
							<el-link type="success" @click="toReg">使用者註冊</el-link>
						</div>
					</el-col>
					<el-col :span="12">
						<div class="grid-content bg-purple-dark">
							<el-link type="success">忘記密碼</el-link>
						</div>
					</el-col>
				</el-row>
			</el-form-item>
		</el-form>
	</div>
</template>

<script>
	export default {
		data() {
			return {
				ruleForm: {
					uname: '',
					pwd: ''
				}
			};
		},
		methods: {
			toReg() {
				this.$router.push({
					path: '/Reg'
				});
			},
			doLogin() {
				let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
				console.log(url)
				this.axios.get(url, {
					params: this.ruleForm
				}).then((response) => {
					console.log(response);
					if (response.data.code == 1) {
						this.$message({
							showClose:true,
							message:response.data.msg,
							type: 'success'
						});
					} else {
						this.$message({
							showClose:true,
							message:response.data.msg,
							type: 'error'
						});
					}

				}).catch((response) => {
					console.log(response);
				});
			}
		}
	}
</script>

<style scoped>
	.login-wrap {
		box-sizing: border-box;
		width: 100%;
		height: 100%;
		padding-top: 10%;
		background-image: url();
		/* background-color: #112346; */
		background-repeat: no-repeat;
		background-position: center right;
		background-size: 100%;
	}

	.login-container {
		border-radius: 10px;
		margin: 0px auto;
		width: 350px;
		padding: 30px 35px 15px 35px;
		background: #fff;
		border: 1px solid #eaeaea;
		text-align: left;
		box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
	}

	.title {
		margin: 0px auto 40px auto;
		text-align: center;
		color: #505458;
	}
</style>

Reg.vue

<template>
	<div class="login-wrap">
		<el-form :model="ruleForm" label-width="100px" class="demo-ruleForm login-container">
			<h3 style="text-align: center;">使用者註冊</h3>
			<el-form-item label="使用者名稱">
				<el-input v-model="ruleForm.uname"></el-input>
			</el-form-item>
			<el-form-item label="密碼">
				<el-input type="password" v-model="ruleForm.pwd"></el-input>
			</el-form-item>
			<el-form-item>
				<el-row>
					<el-col :span="24">
						<div class="grid-content bg-purple-dark">
							<el-button type="primary" style="width:100% ;">提交</el-button>
						</div>
					</el-col>
				</el-row>
				<el-row>
					<el-col :span="12">
						<div class="grid-content bg-purple-dark">
							 <el-link type="success" @click="toLogin">使用者登入</el-link>
						</div>
					</el-col>
					<el-col :span="12">
						<div class="grid-content bg-purple-dark">
							 <el-link type="success">忘記密碼</el-link>
						</div>
					</el-col>
				</el-row>
			</el-form-item>
		</el-form>
	</div>
</template>

<script>
	export default{
		data(){
			return{
				ruleForm:{
					uname:'',
					pwd:''
				}
			};
		},
		methods:{
			toLogin(){
				this.$router.push({
					path:'/Login'
				});
			}
		}
	}
</script>

<style scoped>
	.login-wrap {
		box-sizing: border-box;
		width: 100%;
		height: 100%;
		padding-top: 10%;
		background-image: url();
		/* background-color: #112346; */
		background-repeat: no-repeat;
		background-position: center right;
		background-size: 100%;
	}

	.login-container {
		border-radius: 10px;
		margin: 0px auto;
		width: 350px;
		padding: 30px 35px 15px 35px;
		background: #fff;
		border: 1px solid #eaeaea;
		text-align: left;
		box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
	}

	.title {
		margin: 0px auto 40px auto;
		text-align: center;
		color: #505458;
	}
</style>

App.vue

<template>
  <div id="app">
    <router-view/>
  </div>
</template>

<script>
export default {
  name: 'App'
}
</script>

<style> 
html,
body {
    width: 100%;
    height: 100%;
    box-sizing: border-box;
    padding: 0px;
    margin: 0px;
}
#app {
    font-family: "Avenir", Helvetica, Arial, sans-serif;
    -webkit-font-smoothing: antialiased;
    -moz-osx-font-smoothing: grayscale;
    color: #2c3e50;
    widows: 100%;
    height: 100%;
}
</style>

main.js

// The Vue build version to load with the `import` command
// (runtime-only or standalone) has been set in webpack.base.conf with an alias.
import Vue from 'vue'
import 'element-ui/lib/theme-chalk/index.css'
import App from './App'
import router from './router'
import ElementUI from 'element-ui'
import axios from '@/api/http'
import VueAxios from 'vue-axios'

Vue.use(ElementUI)
Vue.use(VueAxios,axios)
Vue.config.productionTip = false

/* eslint-disable no-new */
new Vue({
	el: '#app',
	router,
	components: {
		App
	},
	template: '<App/>'
})

後臺

在這裡插入圖片描述

biz層

UserBiz

package com.tt.vue.biz;

import com.tt.vue.entity.User;

public interface UserBiz {
	public User login(User user);
}

dao層

UserDao

package com.tt.vue.dao;

import java.util.HashMap;

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

import org.hibernate.HibernateException;
import org.hibernate.Session;
import org.springframework.orm.hibernate5.HibernateCallback;

import com.tt.base.dao.BaseDao;
import com.tt.vue.entity.User;
import com.tt.vue.util.PageBean;
import com.tt.vue.util.StringUtils;

public class UserDao extends BaseDao {

	private static final long serialVersionUID = -2644908989655094673L;

	public List<User> list(User user,PageBean pageBean) {
		BaseDao obj = this;
		return (List<User>) this.getHibernateTemplate().execute(new HibernateCallback<List<User>>() {
			@Override
			public List<User> doInHibernate(Session session) throws HibernateException {
				String hql = "from User where 1=1 ";
				Map<String, Object> map = new HashMap<>();
				String uname = user.getUname();
				String pwd = user.getPwd();
				if (StringUtils.isNotBlank(uname)) {
					hql += " and uname = :uname";
					map.put("uname", uname);
				}
				if (StringUtils.isNotBlank(pwd)) {
					hql += " and pwd = :pwd";
					map.put("pwd", pwd);
				}
				return obj.executeQuery(session, hql, map, pageBean);
			}
		});
	}
}

實體列

User

package com.tt.vue.entity;

public class User {
	private String uname;
	private String pwd;

	public String getUname() {
		return uname;
	}

	public void setUname(String uname) {
		this.uname = uname;
	}

	public String getPwd() {
		return pwd;
	}

	public void setPwd(String pwd) {
		this.pwd = pwd;
	}

	@Override
	public String toString() {
		return "User [uname=" + uname + ", pwd=" + pwd + "]";
	}

	public User(String uname, String pwd) {
		super();
		this.uname = uname;
		this.pwd = pwd;
	}

	public User() {
		super();
	}

}

User.hbm.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
	<class table="t_vue_user" name="com.zking.vue.entity.User">
		<id name="uname" type="java.lang.String" column="uname">
			<generator class="assigned"></generator>
		</id>
		
		<property name="pwd" type="java.lang.String" column="pwd"></property>
	</class>
</hibernate-mapping>

web層

UserAction


package com.tt.vue.web;

import java.util.HashMap;

import java.util.Map;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.opensymphony.xwork2.ModelDriven;
import com.tt.base.web.BaseAction;
import com.tt.vue.biz.UserBiz;
import com.tt.vue.entity.User;
import com.tt.vue.util.JsonData;
import com.tt.vue.util.JwtUtils;
import com.tt.vue.util.PageBean;
import com.tt.vue.util.ResponseUtil;
import com.tt.vue.util.StringUtils;

public class UserAction extends BaseAction implements ModelDriven<User>{

	private UserBiz userBiz;
	private User user = new User();

	public UserBiz getUserBiz() {
		return userBiz;
	}

	public void setUserBiz(UserBiz userBiz) {
		this.userBiz = userBiz;
	}
	 
	public String login() {
		ObjectMapper om = new ObjectMapper();
		JsonData jsonData = null;
		try {
			if(StringUtils.isBlank(user.getUname()) || StringUtils.isBlank(user.getPwd())) {
				jsonData = new JsonData(0, "使用者或者密碼為空", user);
			}else {
				User u = this.userBiz.login(user);
//				Map<String, Object> claims = new HashMap<String, Object>();
//				claims.put("uname",user.getUname());
//				claims.put("pwd", user.getPwd());
//				String jwt = JwtUtils.createJwt(claims, JwtUtils.JWT_WEB_TTL);
//				response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);
				jsonData = new JsonData(1, "登入成功", u);
			}
		} catch (Exception e) {
			e.printStackTrace();
			jsonData = new JsonData(0, "使用者或者密碼錯誤", user);
		}finally {
			try {
				ResponseUtil.write(response, om.writeValueAsString(jsonData));
			} catch (Exception e) {
				e.printStackTrace();
			}
		}
		
		return null;
	}
	
	public String getAsyncData() {
		ObjectMapper om = new ObjectMapper();
		try {
			Thread.sleep(6000);
			ResponseUtil.write(response, om.writeValueAsString("http://www.javaxl.com"));
		} catch (Exception e) {
			e.printStackTrace();
		}
		return null;
	}

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

過濾器CorsFilter.java


package com.tt.vue.util;

import java.io.IOException;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

/**
 * 配置tomcat允許跨域訪問
 * 
 * @author Administrator
 *
 */
public class CorsFilter implements Filter {

	@Override
	public void init(FilterConfig filterConfig) throws ServletException {
	}

	// @Override
	// public void doFilter(ServletRequest servletRequest, ServletResponse
	// servletResponse, FilterChain filterChain)
	// throws IOException, ServletException {
	// HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;
	//
	// // Access-Control-Allow-Origin就是我們需要設定的域名
	// // Access-Control-Allow-Headers跨域允許包含的頭。
	// // Access-Control-Allow-Methods是允許的請求方式
	// httpResponse.addHeader("Access-Control-Allow-Origin", "*");// *,任何域名
	// httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT,
	// DELETE");
	// // httpResponse.setHeader("Access-Control-Allow-Headers", "Origin,
	// // X-Requested-With, Content-Type, Accept");
	//
	// // 允許請求頭Token
	// httpResponse.setHeader("Access-Control-Allow-Headers",
	// "Origin,X-Requested-With, Content-Type, Accept, Token");
	// HttpServletRequest req = (HttpServletRequest) servletRequest;
	// System.out.println("Token=" + req.getHeader("Token"));
	// if("OPTIONS".equals(req.getMethod())) {
	// return;
	// }
	//
	//
	// filterChain.doFilter(servletRequest, servletResponse);
	// }

	@Override
	public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
			throws IOException, ServletException {
		HttpServletResponse resp = (HttpServletResponse) servletResponse;
		HttpServletRequest req = (HttpServletRequest) servletRequest;

		// Access-Control-Allow-Origin就是我們需要設定的域名
		// Access-Control-Allow-Headers跨域允許包含的頭。
		// Access-Control-Allow-Methods是允許的請求方式
		resp.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
		resp.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
		// resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With,
		// Content-Type, Accept");
		// 允許客戶端,發一個新的請求頭jwt
		resp.setHeader("Access-Control-Allow-Headers", "Origin,X-Requested-With, Content-Type, Accept, jwt");

		// 允許客戶端,處理一個新的響應頭jwt
		resp.setHeader("Access-Control-Expose-Headers", "jwt");
		// String sss = resp.getHeader("Access-Control-Expose-Headers");
		// System.out.println("sss=" + sss);

		// 允許請求頭Token
		// httpResponse.setHeader("Access-Control-Allow-Headers","Origin,X-Requested-With,
		// Content-Type, Accept, Token");
		// System.out.println("Token=" + req.getHeader("Token"));

		if ("OPTIONS".equals(req.getMethod())) {// axios的ajax會發兩次請求,第一次提交方式為:option,直接返回即可
			return;
		}
		filterChain.doFilter(servletRequest, servletResponse);
	}

	@Override
	public void destroy() {

	}
}

JwtFilter.java

package com.tt.vue.util;

import java.io.IOException;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import javax.servlet.Filter;
import javax.servlet.FilterChain;
import javax.servlet.FilterConfig;
import javax.servlet.ServletException;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import io.jsonwebtoken.Claims;

/**
 * * JWT驗證過濾器,配置順序 :CorsFilte-->JwtFilter-->struts2中央控制器
 * 
 * @author Administrator
 *
 */
public class JwtFilter implements Filter {

	// 排除的URL,一般為登陸的URL(請改成自己登陸的URL)
	private static String EXCLUDE = "^/vue/userAction_login\\.action?.*$";

	private static Pattern PATTERN = Pattern.compile(EXCLUDE);

	private boolean OFF = true;// true關閉jwt令牌驗證功能

	@Override
	public void init(FilterConfig filterConfig) throws ServletException {
	}

	@Override
	public void destroy() {
	}

	@Override
	public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
			throws IOException, ServletException {
		HttpServletRequest req = (HttpServletRequest) request;
		HttpServletResponse resp = (HttpServletResponse) response;
		String path = req.getServletPath();
		if (OFF || isExcludeUrl(path)) {// 登陸直接放行
			chain.doFilter(request, response);
			return;
		}

		// 從客戶端請求頭中獲得令牌並驗證
		String jwt = req.getHeader(JwtUtils.JWT_HEADER_KEY);
		Claims claims = this.validateJwtToken(jwt);
		if (null == claims) {
			// resp.setCharacterEncoding("UTF-8");
			resp.sendError(403, "JWT令牌已過期或已失效");
			return;
		} else {
			String newJwt = JwtUtils.copyJwt(jwt, JwtUtils.JWT_WEB_TTL);
			resp.setHeader(JwtUtils.JWT_HEADER_KEY, newJwt);
			chain.doFilter(request, response);
		}
	}

	/**
	 * 驗證jwt令牌,驗證通過返回宣告(包括公有和私有),返回null則表示驗證失敗
	 */
	private Claims validateJwtToken(String jwt) {
		Claims claims = null;
		try {
			if (null != jwt) {
				claims = JwtUtils.parseJwt(jwt);
			}
		} catch (Exception e) {
			e.printStackTrace();
		}
		return claims;
	}

	/**
	 * 是否為排除的URL
	 * 
	 * @param path
	 * @return
	 */
	private boolean isExcludeUrl(String path) {
		Matcher matcher = PATTERN.matcher(path);
		return matcher.matches();
	}

	// public static void main(String[] args) {
	// String path = "/sys/userAction_doLogin.action?username=zs&password=123";
	// Matcher matcher = PATTERN.matcher(path);
	// boolean b = matcher.matches();
	// System.out.println(b);
	// }

}


其他的後臺程式碼就先不放上來了,解決一個maven專案沒有tomcat的問題,右鍵在這裡插入圖片描述

然後target檔案下面會產生war檔案。把war拷到tomcat檔案下面的webapps下面就可以了

相關文章