SPA项目开发之登录

需要注意的问题
1、 ajax 跨域问题
2、 A xios 的 get/post 的区别( qs )
3、 简化axios 使用的书写
4、 T his 指针带来的变量污染,需要利用箭头函数来解决;

先创建一个新项目再说

  vue init webpack  cpc

cd cpc

npm install

npm install element-ui -S  

config --> index.js 8088

npm run dev

npm install axios -S

npm install qs -S  

npm install vue-axios -S

说明:
npm install element-ui -S 安装 element-ui 模块

npm install axios -S axios 是vue2提倡使用的轻量版的ajax。它是基于promise的HTTP库。 它会从浏览器中创建XMLHttpRequests,与Vue配合使用非常好。

npm install qs -S qs 这是一个用来转换格式的东西 例如:将{a:‘b’,c:‘d’}转换成’a=b&c=d’

npm install vue-axios -S vue-axios是在axios基础上扩展的插件,在Vue.prototype原型上扩展了$http等属性,可以更加方便的使用axios

向项目中的src目录导入一个工具包:
在这里插入图片描述
作用主要是用来简化开发的

action.js
这个文件是请求映射的js文件,写成这样如果以后要改懂请求地址,只要修改这个文件就ojbk了。

/**
 * 对后台请求的地址的封装,URL格式如下:
 * 模块名_实体名_操作
 */
export default {
	'SERVER': 'http://localhost:8080/T216_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

http.js 中相当于简化了我们的开发,它帮我们已经引入好了 axios qs action 这三个模块,我们只需要在main.js中直接映入就ok了。(main.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;

main.js
这是程序的入口文件,将需要用到的组件加载(Vue.use(xxx))到了Vue中去。Vue实例只需要通过this.xxx就能访问到了。( 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 App from './App'
import router from './router'
//这里是发送请求要引入的组件
import axios from '@/api/http'
import VueAxios from 'vue-axios'
//添加elementUI
import ElementUI from 'element-ui'
 //这里导入 element-ui的css文件
import 'element-ui/lib/theme-chalk/index.css'

//加载 VueAxios 和 axios
Vue.use(VueAxios,axios)

//加载 ElementUI
Vue.use(ElementUI)

Vue.config.productionTip = false
/* eslint-disable no-new */
new Vue({
  el: '#app',
  //这里已经帮我们挂载好了路由器
  router,
  components: { App },
  template: '<App/>'
})

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
    }
  ]
})

注册页面

<template>
  <div class="login-wrap">
    <el-form :model="ruleForm" status-icon :rules="rules" ref="ruleForm" label-width="100px" class="demo-ruleForm login-container">
      <h3 style="text-align: center;">用户注册</h3>
      <el-form-item label="用户名">
        <el-input v-model.number="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="info">忘记密码</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>

登录页面

<template>
  <div class="login-wrap">
    <el-form :model="ruleForm" ref="ruleForm" label-width="100px" class="demo-ruleForm login-container">
      <h3 style="text-align: center;">用户登录</h3>
      <el-form-item label="用户名">
        <el-input v-model.number="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="info">忘记密码</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;
        this.axios.get(url, {
          //params是发生到后端的json数据
          params: this.ruleForm
        }).then((response) => {
          //这是请求成功的回掉
          if (response.data.code == 1) {
            this.$message({
              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>

post的方式请求的登录

	//加载的是 http.js文件所以这里直接 this.ruleForm 就Ojbk了
	 axios.post(url, this.ruleForm).then((response) => {
          if (response.data.code == 1) {
            this.$message({
              message: '登录成功,牛逼',
              type: 'success'
            });
          } else {
            this.$message.error(response.data.msg);
          }
        }).catch((response) => {
          console.log(response);
        });

特别说明:

如果main.js中引入是:

import axios from 'axios ' 

那么你就要引入 qs 这个模块,在登录.vue文件中的js代码中中添加 如下代码

import qs from 'qs'

发post的请求的时候要调用 qs.stringify(this.ruleForm) 来进行格式转换 (如果不转换后端将获取不到数据)

	  axios.post(url, qs.stringify(this.ruleForm)).then((response) => {
          console.log(response);
          if (response.data.code == 1) {
            this.$message({
              message: '登录成功,牛逼',
              type: 'success'
            });
          } else {
            this.$message.error(response.data.msg);
          }
        }).catch((response) => {
          console.log(response);
        });

PS:如果你引入的不是 http.js,那么请也将 let url = this.axios.urls.SYSTEM_USER_DOLOGIN; 改改呀

解释:
因为POST提交的参数的格式是Request Payload,这样后台取不到数据的
解决方案:使用qs.js库,将{a:‘b’,c:‘d’}转换成’a=b&c=d’ 如果引入http.js你也看到了,里面已经引入了qs,在发请求的时候会自动将格式转好。而如果你用的是 **import axios from ‘axios’**那就得安装上面的玩,不过我不推荐你这么玩。就是的啦。

如果希望这样调用请求方法 axios.get 那么你之间在当前页面引入组件就完事了:

<script>
  import axios from 'axios'
  import VueAxios from 'vue-axios'
  import action from '@/api/action'
  import qs from 'qs'
  .....
</script>

公共模块没有必要这么做,一般是页面私有的模块这么去引入。

解决跨越问题

会一直报错:“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 #允许特定的域名访问

/**
 * 配置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-Headers", "Origin, X-Requested-With, Content-Type, Accept");
		httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
		filterChain.doFilter(servletRequest, servletResponse);
	}

	@Override
	public void destroy() {

	}
}

web.xml中配置一下:

	<!-- 解决cors跨域问题过滤器 -->
	<filter>
		<filter-name>corsFilter</filter-name>
		<filter-class>com.cpc.vue.util.CorsFilter</filter-class>
	</filter>
	<filter-mapping>
		<filter-name>corsFilter</filter-name>
		<url-pattern>/*</url-pattern>
	</filter-mapping>
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值