一.跨域的概述
1.1.概述
- 跨域(Cross-Origin)指的是在浏览器上执行的 Web 应用程序试图访问不同域名下的资源或向不同域名的服务器发送请求时所面临的限制。浏览器出于安全考虑实施了同源策略,即只允许网页上的脚本访问同一源的资源,所谓同源指的是协议、域名和端口号完全相同。
- 当浏览器上的 Web 应用程序试图跨域请求资源时,浏览器会阻止这些请求,以防止恶意网站窃取用户的数据。为了解决跨域问题,浏览器提供了一些机制如跨域资源共享(CORS)、JSONP、代理等。
- 跨域资源共享(CORS)是最常用的跨域解决方案之一。通过在服务器端设置响应头部,允许特定的源(域、协议、端口)来访问资源。通过在请求头部中添加 Origin 字段,浏览器可以告知服务器访问来源。服务器在响应中添加 Access-Control-Allow-Origin 头部,明确指定允许的来源,从而解决跨域访问问题。
- JSONP是一种通过<script>标签来实现的跨域方式。在请求中使用<script>标签加载远程JS文件时,可以通过回调函数的形式获取数据。由于<script>标签没有同源限制,可以跨域加载,因此可以用于解决跨域问题。
- 代理是另一种解决跨域问题的方法。通过在同一域名下的服务器上设置代理服务,实现对跨域请求的转发。当浏览器请求某个接口时,先将请求发送到同源服务器上的代理服务,再由代理服务转发到目标服务器上,获取结果后再返回给浏览器。
总结来说,跨域是浏览器的一种安全机制,限制网页对不同域名资源的访问。通过使用跨域资源共享、JSONP、代理等技术,可以解决跨域问题,实现不同域名之间的数据交互。
1.2.特点
跨域的主要特点如下:
- 安全性:跨域是浏览器的一项安全机制,旨在防止恶意网站通过跨域请求来获取用户的敏感数据。
- 同源策略:同源策略要求网页上的脚本只能访问相同源的资源,同源指的是协议、域名和端口号完全相同。这种限制确保了网站只能访问自身域名下的资源,从而保护了用户的隐私和安全。
- 跨域解决方案:为了克服同源策略带来的限制,浏览器提供了一些跨域解决方案,如跨域资源共享(CORS)、JSONP、代理等。这些解决方案可以在一定程度上实现不同域名之间的数据交互。
- CORS:跨域资源共享(CORS)是最常用的跨域解决方案之一。通过在服务器端设置响应头部,允许特定的源(域、协议、端口)来访问资源。
- JSONP:JSONP是一种通过<script>标签来实现的跨域方式。通过回调函数的形式获取数据,可以跨域加载远程JS文件。
- 代理:代理是另一种解决跨域问题的方法。通过在同一域名下的服务器上设置代理服务,实现对跨域请求的转发。
- 部署和配置:在使用跨域解决方案时,需要在服务器端进行相应的配置和部署工作,以确保跨域请求能够正常进行。
跨域具有安全性、同源策略限制、跨域解决方案、配置和部署等特点。了解和掌握跨域相关的知识和技术将有助于开发人员解决跨域问题,实现跨域数据交互。
如何跨域:
需要进行前后端访问,及数据进行交互就需要解决跨域的问题。
以下是跨域问题的体现,有该错误就说明是跨域问题
如图 :
需要解决呢,在maven项目中的创建一个过滤器,名为: CorsFilter
代码如下 :
package com.junlinyi.ssm.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; HttpServletRequest req = (HttpServletRequest) servletRequest; // Access-Control-Allow-Origin就是我们需要设置的域名 // Access-Control-Allow-Headers跨域允许包含的头。 // Access-Control-Allow-Methods是允许的请求方式 httpResponse.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名 httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE"); //允许客户端发一个新的请求头jwt httpResponse.setHeader("Access-Control-Allow-Headers","responseType,Origin,X-Requested-With, Content-Type, Accept, jwt"); //允许客户端处理一个新的响应头jwt httpResponse.setHeader("Access-Control-Expose-Headers", "jwt,Content-Disposition"); //httpResponse.setHeader("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept"); //httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE"); // axios的ajax会发两次请求,第一次提交方式为:option,直接返回即可 if ("OPTIONS".equals(req.getMethod())) { return; } filterChain.doFilter(servletRequest, servletResponse); } @Override public void destroy() { } }
如何在maven项目中的 web.xml 文件中进行配置该过滤器
配置如下 :
<!--CrosFilter跨域过滤器--> <filter> <filter-name>corsFilter</filter-name> <filter-class>com.junlinyi.ssm.util.CorsFilter2</filter-class> </filter> <filter-mapping> <filter-name>corsFilter</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
完成以上就可以解决跨域的问题,之后就可以进行前后端的访问,进行数据交互
二.ElementUI
2.1. 导入
第一步:使用CMD命令窗口,并跳转到指定工作目录下创建项目
输入以下命令来创建项目 :
vue init webpack spa
注 : 其中 spa 是项目名称,可以自己进行修改项目名称。
关于项目的创建及搭建在我博客中的项目搭建的构建中:
一篇让你使用vue-cli搭建SPA项目
第二步:使用CMD命令窗口,并跳转到指定项目的目录下面
如 : xx spa (其中spa是项目名称已是要进入的目录名称)
第三步:使用命令npm install element-ui -S,添加Element-UI模块到项目中
操作如图:
最后,打开项目的package.json文件可查看具体添加模块信息。
如图所示 :
2.2.搭建
在项目中的src目录下创建views目录(该目录用于存放vue组件)。
添加Element-UI模块到项目中后,在项目的src目录中找main.js中引入element-ui模块
需要引入的代码:
import ElementUI from 'element-ui'
import 'element-ui/lib/theme-chalk/index.css'
Vue.use(ElementUI)
注 : 导入依赖需要按照顺序要求进行
如图:
所有代码: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' // 1.导入ElementUI依赖 import ElementUI from 'element-ui' // 2.导入ElementUI样式,避免后期打包样式不同,要放在import App from './App';之前 import 'element-ui/lib/theme-chalk/index.css' import App from './App' import router from './router' // 3增加ElementUI实例 Vue.use(ElementUI) Vue.config.productionTip = false import axios from '@/api/http' import VueAxios from 'vue-axios' Vue.use(VueAxios,axios) /* eslint-disable no-new */ new Vue({ el: '#app', router, components: { App }, template: '<App/>' })
2.3.页面
在views目录下创建一个Login.vue组件,登入页面的显示,代码如下:
<template> <div id="app"> <template> <div class="login-wrap"> <el-form class="login-container"> <h1 class="title">用户登录</h1> <el-form-item label=""> <el-input type="text" v-model="username" placeholder="登录账号" autocomplete="off"></el-input> </el-form-item> <el-form-item label=""> <el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off"></el-input> </el-form-item> <el-form-item> <el-button type="primary" style="width:100%;" @click="doSubmit()">提交</el-button> </el-form-item> <el-row style="text-align: center;margin-top:-10px"> <el-link type="primary">忘记密码</el-link> <el-link type="primary" @click="gotoRegister()">用户注册</el-link> </el-row> </el-form> </div> </template> </div> </template> <script> import axios from 'axios' import qs from 'qs' export default { name: 'Login', data () { return { username:'', password:'' } } ,methods:{ gotoRegister(){ // router-like相当于a // location.href 相当于this.$router.push this.$router.push('/Register'); }, doSubmit(){ let url =this.axios.urls.SYSTEM_USER_DOLOGIN; let params ={ username:this.username, password:this.password } // $.ajax then相当于success // axios.get(url,{params:params}).then(r=>{ // console.log(r); // if(r.data.success){ // this.$message({ // message:r.data.msg, // type: 'success' // }); // }else{ // this.$message.error(r.data.msg); // } // }).cath(e=>{ // }) this.axios.post(url,params).then(r => { console.log(r); //如果携带的参数数据跟后端数据对应正确,说明登入成功,提示 if (r.data.success) { this.$message({ showClose: true, message: r.data.msg, type: 'success' }); } else { //如果携带的参数数据跟后端数据对应错误,说明登入失败,提示 this.$message.error(r.data.msg); } }).catch(e => { console.log(e); }); } } } </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>
然后在项目路径下面,输入以下命令开启项目。
开启命令 : npm run pev
开起后访问指定路径在浏览器中进行访问,如 :http://localhost:8080
效果如图 :
三.数据交互
3.1.安装相关模块
3.1.1安装模块
在我们的maven项目中,需要前后端请求的话,需要在maven项目中安装ajax的模块
在前端的spa项目中,在本地的根本路径下,Win+R,输入cmd,打开cmd窗口。
输入以下命令安装所需模块 :
npm i axios -S
npm i qs-S
操作命令如图 :
axios是vue2提倡使用的轻量版的ajax。它是基于promise的HTTP库。它会从浏览器中创建XMLHttpRequests,与Vue配合使用非常好。
冗余代码的模块
在此,我们前端的代码会有请求路径及引入模块等等的很多冗余代码。
为解决这一问题,我们只需下载一个vue-axios的引用整合模块。
在项目中安装之后,对于后期项目的维护有极大的帮助和很大的提升项目开发的效率
在前端的spa项目中,在本地的根本路径下,Win+R,输入cmd,打开cmd窗口。
输入以下命令安装所需模块 :
npm i vue-axios -S
vue-axios是在axios基础上扩展的模块,在Vue.prototype原型上扩展了$http等属性,可以更加方便的使用axios。
3.1.2查看模块
在前端项目的package.json的文件中看到以下,就说明安装模块已经完成
3.1.3.引用模块
安装后就在项目中进行引用,添加axios的全局配置,创建一个actio.js
/** * 对后台请求的地址的封装,URL格式如下: * 模块名_实体名_操作 */ export default { 'SERVER': 'http://localhost:8080/ssm', //服务器地址 'SYSTEM_USER_DOLOGIN': '/user/userLogin', //登陆请求 'SYSTEM_USER_DOREG': '//user/userRegister', //注册请求 'getFullPath': k => { //获得请求的完整地址,用于mockjs测试时使用 return this.SERVER + this[k]; } }
action.js(针对后台请求接口的封装定义)和http.js(针对axios的全局配置)两个文件。
创建http.js,用于vue项目对axios的全局配置
/** * 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) { return config; }, function(error) { return Promise.reject(error); }); // 响应拦截器 axios.interceptors.response.use(function(response) { 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;
之后在前端项目中,关联相关的创建及js文件进行应用。
在前端的项目中main.js文件中引入vue-axios 模块
// 在main.js文件中引入vue-axios模块 import axios from '@/api/http' import VueAxios from 'vue-axios' / 在main.js文件中引入vue-axios模块 Vue.use(VueAxios,axios)
Axios是一个基于Promise 用于浏览器和 nodejs 的 HTTP 客户端,本质上也是对原生XHR的封装,只不过它是Promise的实现版本,符合最新的ES规范。
3.2. axios的get请求
在我们的登入组件中进行axios的get请求,编写Login.vue的代码
将script标签的代码修改为以下代码进行get请求
<script> import axios from 'axios' import qs from 'qs' export default { name: 'Login', data () { return { username:'', password:'' } } ,methods:{ gotoRegister(){ // router-like相当于a // location.href 相当于this.$router.push this.$router.push('/Register'); }, doSubmit(){ let url =this.axios.urls.SYSTEM_USER_DOLOGIN; let params ={ username:this.username, password:this.password } $.ajax then相当于success axios.get(url,{params:params}).then(r=>{ console.log(r); if(r.data.success){ this.$message({ message:r.data.msg, type: 'success' }); }else{ this.$message.error(r.data.msg); } }).cath(e=>{ }) } } } </script>
在我们的后端项目中,有一个视图解析器来处理我们前端发过来的请求进行处理
package com.junlinyi.ssm.controller; import com.junlinyi.ssm.service.IUserService; import com.junlinyi.ssm.util.JsonResponseBody; import com.junlinyi.ssm.util.PageBean; import com.junlinyi.ssm.vo.UserVo; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.HashMap; import java.util.List; import java.util.Map; import com.junlinyi.ssm.jwt.*; @Controller @RequestMapping("/user") public class UserController { @Autowired private IUserService userService; @RequestMapping("/userLogin") @ResponseBody public JsonResponseBody<?> userLogin(UserVo userVo, HttpServletResponse response){ if(userVo.getUsername().equals("admin")&&userVo.getPassword().equals("123")){ //私有要求claim // Map<String,Object> json=new HashMap<String,Object>(); // json.put("username", userVo.getUsername()); //生成JWT,并设置到response响应头中 // String jwt=JwtUtils.createJwt(json, JwtUtils.JWT_WEB_TTL); // response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt); return new JsonResponseBody<>("用户登陆成功!",true,0,null); }else{ return new JsonResponseBody<>("用户名或密码错误!",false,0,null); } } @RequestMapping("/queryUserPager") @ResponseBody public JsonResponseBody<List<Map<String,Object>>> queryUserPager(UserVo userVo, HttpServletRequest request){ try { PageBean pageBean=new PageBean(); pageBean.setRequest(request); List<Map<String, Object>> users = userService.queryUserPager(userVo, pageBean); return new JsonResponseBody<>("OK",true,pageBean.getTotal(),users); } catch (Exception e) { e.printStackTrace(); return new JsonResponseBody<>("分页查询用户信息失败!",false,0,null); } } }
注 : ( 其中的导包需要根据自己的项目包的结构进行导包方可 )
在前端项目的根本路径下,Win+R,输入cmd,打开cmd窗口。
输入命令开启项目 :
npm run dev
并且在后端的maven项目,开启服务器(开启项目),可以进行访问才可
之后在指定的请求地址进行访问,显示前端的页面(视图或者说组件)
展现的效果如图 :
3.3. axios的post请求
使用pos请求一并也进行资源的整合
之前我们把需要的应用模块及组件配置都已经完成,现在我们只需要编写我们的登入组件。
将名为 : Login.vue 登入组件的所有代码为以下代码即可。
<template> <div id="app"> <template> <div class="login-wrap"> <el-form class="login-container"> <h1 class="title">用户登录</h1> <el-form-item label=""> <el-input type="text" v-model="username" placeholder="登录账号" autocomplete="off"></el-input> </el-form-item> <el-form-item label=""> <el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off"></el-input> </el-form-item> <el-form-item> <el-button type="primary" style="width:100%;" @click="doSubmit()">提交</el-button> </el-form-item> <el-row style="text-align: center;margin-top:-10px"> <el-link type="primary">忘记密码</el-link> <el-link type="primary" @click="gotoRegister()">用户注册</el-link> </el-row> </el-form> </div> </template> </div> </template> <script> import axios from 'axios' import qs from 'qs' export default { name: 'Login', data () { return { username:'', password:'' } } ,methods:{ gotoRegister(){ // router-like相当于a // location.href 相当于this.$router.push this.$router.push('/Register'); }, doSubmit(){ let url =this.axios.urls.SYSTEM_USER_DOLOGIN; let params ={ username:this.username, password:this.password } // $.ajax then相当于success // axios.get(url,{params:params}).then(r=>{ // console.log(r); // if(r.data.success){ // this.$message({ // message:r.data.msg, // type: 'success' // }); // }else{ // this.$message.error(r.data.msg); // } // }).cath(e=>{ // }) this.axios.post(url,params).then(r => { console.log(r); //如果携带的参数数据跟后端数据对应正确,说明登入成功,提示 if (r.data.success) { this.$message({ showClose: true, message: r.data.msg, type: 'success' }); } else { //如果携带的参数数据跟后端数据对应错误,说明登入失败,提示 this.$message.error(r.data.msg); } }).catch(e => { console.log(e); }); } } } </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>
展现的效果已是一样,并且将代码减少提高开发效率。
四.注册功能
在前端创建组件 Register.vue
<template> <div class="login-wrap"> <el-form class="login-container"> <h1 class="title">用户注册</h1> <el-form-item label=""> <el-input type="text" v-model="username" placeholder="输入账号" autocomplete="off"></el-input> </el-form-item> <el-form-item label=""> <el-input type="password" v-model="password" placeholder="输入密码" show-password autocomplete="off"></el-input> </el-form-item> <el-form-item label="账号名称" label-width="80px" style="margin-left: 5px;" prop="name"> <el-input v-model="realname"></el-input> </el-form-item> <el-form-item label="性别" style="margin-left: 20px;"> <el-radio-group v-model="sex"> <el-radio label="1">男</el-radio> <el-radio label="2">女</el-radio> <el-radio label="3">人妖</el-radio> </el-radio-group> </el-form-item> <el-form-item label="身 份 证" label-width="80px" style="margin-left: 0px;" prop="name"> <el-input v-model="idcard"></el-input> </el-form-item> <el-input type="textarea" :rows="2" placeholder="请输入地址" v-model="address"> </el-input> <el-form-item style="margin-top: 30px;"> <el-button type="primary" style="width:100%;" @click="Register()">注 册</el-button> </el-form-item> <el-row style="text-align: center;margin-top:-10px"> <el-link type="primary">忘记密码</el-link> <el-link type="primary" @click="Login()">用户登入</el-link> </el-row> </el-form> </div> </template> <script> export default { name: 'Register', data() { return { username: "", password: "", realname: '', sex: 0, idcard: '', address: '', msg: '嗨!嗨!嗨!', rules: { realname: [{ required: true, message: '请输入账号名称', trigger: 'blur' }, { min: 3, max: 18, message: '长度在 3 到 18 个字符', trigger: 'blur' } ] } } }, methods: { Login() { this.$router.push('/'); }, Register() { let params = { username: this.username, password: this.password, realname: this.realname, sex: this.sex, idcard: this.idcard, address: this.address }; //定义后端都请求地址 var url = this.axios.urls.SYSTEM_USER_DOREG; console.log(params); console.log(url); //以下是post请求及整合资源 //通过qs中的stringify方法进行格式转换 //注意数据是保存到json对象的params属性 this.axios.post(url, params).then(r => { console.log(r); //如果携带的参数数据跟后端数据对应正确,说明登入成功,提示 if (r.data.success) { this.$message({ showClose: true, message: r.data.msg, type: 'success' }); //注册完成后自动进入登入界面 this.$router.push('/'); } else { //如果携带的参数数据跟后端数据对应错误,说明登入失败,提示 this.$message.error(r.data.msg); } }).catch(e => { console.log(e); }); } } } </script> <style scoped> .login-wrap { padding-top: 30px; box-sizing: border-box; width: 100%; height: 100%; 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>
在后端的maven项目中 IUserService 的代码
int insertSelective(User record);
在后端的maven项目中 UserServiceImpl 的代码
@Override public int insertSelective(User record) { return userMapper.insertSelective(record); }
在后端的maven项目中 UserController(控制器) 的代码
@RequestMapping("/userRegister") @ResponseBody public JsonResponseBody<?> userRegister(UserVo userVo, HttpServletResponse response){ //状态新注册默认为0 userVo.setStatus("0"); //因为ID为String类型需要手动设置,当然可以根据自己的需要改为Int类型 userVo.setId("5"); int i = userService.insertSelective(userVo); if(i>0){ return new JsonResponseBody<>("用户注册完成!快去登入吧!",true,0,null); }else{ return new JsonResponseBody<>("用户注册失败!重新输入。",false,0,null); } }
展现效果 :