• SPA项目的登录注册实现,post/get请求以及跨域问题


    目录

    前言

    一. 登录,注册静态页面

    1.1 ElementUI简介

     1.2 基于SPA项目完成登录

    1.2.1 在SPA项目中添加elementui依赖

    1.2.2 在main.js中添加elementui模块

    1.2.3 在src目录下创建views目录,用于存放vue组件

    1.2.4 配置路由

     1.2.5 修改项目端口并启动

    二. 数据交互

    2.1.安装axios

    3.2.axios之get

    3.3.axios之post

    3.4.vue-axios

    3.5 登录注册后端代码

    service层

    controller层

    3.6 效果展示

    三. 跨域问题

    3.1 什么是跨域

    3.2 怎么解决跨域问题


    前言

    上一篇博客分享了SPA项目的构建,接下来,在SPA项目上接着完成登录注册的实现。

    一. 登录,注册静态页面

    1.1 ElementUI简介

    Element-Ul是饿了么前端团队推出的一款基于Vue.js 2.0 的桌面端UI框架。

    Element-Ul是一个ui库,它不依赖于vue。但是却是当前和vue配合做项目开发的一个比较好的ui框架。

    ElementUI的优势包括:

    1. 简单易用:ElementUI提供了一致的API和文档,使开发者能够快速上手并使用组件。它的设计风格简洁明了,易于理解和操作。

    2. 高度可定制:ElementUI的组件具有丰富的配置选项,可以根据项目需求进行灵活的定制。开发者可以自定义样式、主题和功能,以满足不同的设计要求。

    3. 响应式布局:ElementUI的组件支持响应式布局,可以适应不同的屏幕尺寸和设备类型。这使得应用程序在不同的平台上都能提供良好的用户体验。

    4. 多语言支持:ElementUI提供了多语言支持,可以轻松地切换和本地化不同的语言。这对于开发多语言环境下的应用程序非常有用。

    5. 活跃的社区支持:ElementUI拥有庞大的开发者社区,提供了丰富的资源和支持。开发者可以通过社区获取帮助、分享经验和参与贡献,使得开发过程更加顺利和高效。

     1.2 基于SPA项目完成登录

    1.2.1 在SPA项目中添加elementui依赖

    在创建的SPA项目根路径下cmd回车输入以下命令

    npm install element-ui -S

    成功后会在package.json中看到elementui依赖

    1.2.2 在main.js中添加elementui模块

    在项目中src目录下找到main.js ,并在指定位置添加三行代码:

    1. import Vue from 'vue'
    2. // 新添加1
    3. import ElementUI from 'element-ui'
    4. // 新添加2,避免后期打包样式不同,要放在import App from './App';之前
    5. import 'element-ui/lib/theme-chalk/index.css'
    6. import App from './App'
    7. import router from './router'
    8. // 新添加3
    9. Vue.use(ElementUI)
    10. Vue.config.productionTip = false

    重要的事情说三遍:在指定位置!!!在指定位置!!!在指定位置!!!~~~添加三行代码  

    1.2.3 在src目录下创建views目录,用于存放vue组件

    创建用户登录组件login.vue

    1. <template>
    2. <div class="login-wrap">
    3. <el-form class="login-container">
    4. <h1 class="title">用户登录h1>
    5. <el-form-item label="">
    6. <el-input type="text" v-model="username" placeholder="登录账号" autocomplete="off">el-input>
    7. el-form-item>
    8. <el-form-item label="">
    9. <el-input type="password" v-model="password" placeholder="登录密码" autocomplete="off">el-input>
    10. el-form-item>
    11. <el-form-item>
    12. <el-button type="primary" style="width:100%;" @click="doSubmit()">提交el-button>
    13. el-form-item>
    14. <el-row style="text-align: center;margin-top:-10px">
    15. <el-link type="primary">忘记密码el-link>
    16. <el-link type="primary" @click="gotoRegister()">用户注册el-link>
    17. el-row>
    18. el-form>
    19. div>
    20. template>
    21. <script>
    22. export default {
    23. name: 'Login',
    24. data () {
    25. return {
    26. username:'',
    27. password:''
    28. }
    29. },
    30. methods:{
    31. gotoRegister(){
    32. this.$router.push('/Register');
    33. }
    34. }
    35. }
    36. script>
    37. <style scoped>
    38. .login-wrap {
    39. box-sizing: border-box;
    40. width: 100%;
    41. height: 100%;
    42. padding-top: 10%;
    43. background-image: url();
    44. /* background-color: #112346; */
    45. background-repeat: no-repeat;
    46. background-position: center right;
    47. background-size: 100%;
    48. }
    49. .login-container {
    50. border-radius: 10px;
    51. margin: 0px auto;
    52. width: 350px;
    53. padding: 30px 35px 15px 35px;
    54. background: #fff;
    55. border: 1px solid #eaeaea;
    56. text-align: left;
    57. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    58. }
    59. .title {
    60. margin: 0px auto 40px auto;
    61. text-align: center;
    62. color: #505458;
    63. }
    64. style>

     

    创建用户注册组件Register.vue 

    1. <template>
    2. <div class="login-wrap">
    3. <el-form class="login-container">
    4. <h1 class="title">用户注册h1>
    5. <el-form-item label="">
    6. <el-input type="text" v-model="id" placeholder="编号" autocomplete="off">el-input>
    7. el-form-item>
    8. <el-form-item label="">
    9. <el-input type="text" v-model="realname" placeholder="姓名" autocomplete="off">el-input>
    10. el-form-item>
    11. <el-form-item label="">
    12. <el-input type="text" v-model="username" placeholder="注册账号" autocomplete="off">el-input>
    13. el-form-item>
    14. <el-form-item label="">
    15. <el-input type="password" v-model="password" placeholder="注册密码" autocomplete="off">el-input>
    16. el-form-item>
    17. <el-form-item>
    18. <el-button type="primary" style="width:100%;" @click="doSubmit()">提交el-button>
    19. el-form-item>
    20. <el-row style="text-align: center;margin-top:-10px">
    21. <el-link type="primary" @click="gotoLogin()">已有账号,去登录el-link>
    22. el-row>
    23. el-form>
    24. div>
    25. template>
    26. <script>
    27. export default {
    28. name: 'Register',
    29. data () {
    30. return {
    31. id:'',
    32. realname:'',
    33. username:'',
    34. password:''
    35. }
    36. },
    37. methods:{
    38. gotoLogin(){
    39. this.$router.push('/');
    40. },
    41. }
    42. }
    43. script>
    44. <style scoped>
    45. .login-wrap {
    46. box-sizing: border-box;
    47. width: 100%;
    48. height: 100%;
    49. padding-top: 10%;
    50. background-image: url();
    51. /* background-color: #112346; */
    52. background-repeat: no-repeat;
    53. background-position: center right;
    54. background-size: 100%;
    55. }
    56. .login-container {
    57. border-radius: 10px;
    58. margin: 0px auto;
    59. width: 350px;
    60. padding: 30px 35px 15px 35px;
    61. background: #fff;
    62. border: 1px solid #eaeaea;
    63. text-align: left;
    64. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    65. }
    66. .title {
    67. margin: 0px auto 40px auto;
    68. text-align: center;
    69. color: #505458;
    70. }
    71. style>

    在vue组件中,在style标签上添加scoped属性,以表示它的样式作用于当下的模块,很好的实现了样式私有化的目的。 

     

    1.2.4 配置路由

    在 router/index.js 中修改vue项目默认显示路由

    1. import Vue from 'vue'
    2. import Router from 'vue-router'
    3. import HelloWorld from '@/components/HelloWorld'
    4. import Login from '@/views/Login'
    5. import Register from '@/views/Register'
    6. Vue.use(Router)
    7. export default new Router({
    8. routes: [
    9. {
    10. path: '/',
    11. name: 'Login',
    12. component: Login
    13. },{
    14. path: '/Register',
    15. name: 'Register',
    16. component: Register
    17. }
    18. ]
    19. })

     1.2.5 修改项目端口并启动

    在 config/index.js 目录下修改vue项目运行端口

    二. 数据交互

    构建一个Java后台SSM项目,模拟提供一个用户登录的action地址,Vue通过请求指定的用户登录接口。

    2.1.安装axios

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

    题外话:vue.js有著名的全家桶系列:vue-router,vuex, vue-resource,再加上构建工具vue-cli,就是一个完整的vue项目的核心构成。 其中vue-resource是Vue.js的一款插件,它可以通过XMLHttpRequest或JSONP发起请求并处理响应,但在vue更新到2.0之后,作者就宣告不再对vue-resource更新,而是推荐的axios

    安装axios命令:

    npm i axios -S 

    3.2.axios之get

    使用axios的get请求方式,必须将请求参数保存到json对象的params属性中

    用户登录:

    1. <script>
    2. import axios from 'axios'
    3. import qs from 'qs'
    4. export default {
    5. name: 'Login',
    6. data() {
    7. return {
    8. username: '',
    9. password: ''
    10. }
    11. },
    12. methods: {
    13. gotoRegister() {
    14. this.$router.push('/Register');
    15. },
    16. doSubmit() {
    17. let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
    18. let params = {
    19. username: this.username,
    20. password: this.password
    21. };
    22. // get请求
    23. axios.get(url, {
    24. params: params
    25. }).then(r => {
    26. console.log(r);
    27. if (r.data.success) {
    28. this.$message({
    29. message: r.data.msg,
    30. type: 'success'
    31. });
    32. }else{
    33. this.$message.error(r.data.msg);
    34. }
    35. }).catch(e => {
    36. })
    37. }
    38. }
    39. }
    40. script>

    3.3.axios之post

    使用axios的post请求方式,直接将请求参数保存到json对象中即可

    用户注册:

    1. <script>
    2. export default {
    3. name: 'Register',
    4. data () {
    5. return {
    6. id:'',
    7. realname:'',
    8. username:'',
    9. password:''
    10. }
    11. },
    12. methods:{
    13. gotoLogin(){
    14. this.$router.push('/');
    15. },
    16. doSubmit() {
    17. let url = this.axios.urls.SYSTEM_USER_DOREG;
    18. let params = {
    19. id : this.id,
    20. realname: this.realname,
    21. username: this.username,
    22. password: this.password
    23. };
    24. // post请求
    25. this.axios.post(url, params).then(r => {
    26. console.log(r);
    27. if (r.data.success) {
    28. this.$message({
    29. message: r.data.msg,
    30. type: 'success'
    31. });
    32. }else{
    33. this.$message.error(r.data.msg);
    34. }
    35. }).catch(e => {
    36. })
    37. }
    38. }
    39. }
    40. script>

    axios.post提交后台接收不到数据的,因为POST提交的参数的格式是Request Payload

    解决方案:使用qs.js库,将{a:'b',c:'d'}转换成'a=b&c=d'

     下载qs库 npm install qs -S

     在Login.vue组件中导入qs import qs from 'qs'

     通过qs的stringify方法进行格式转换 let str=qs.stringify(params);

    3.4.vue-axios

    Axios是一个基于Promise 用于浏览器和 nodejs 的 HTTP 客户端,本质上也是对原生XHR的封装,只不过它是Promise的实现版本,符合最新的ES规范。

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

    通过vue-axios实现对axios的轻量封装:

    第一步:下载安装vue-axios

    npm i vue-axios -S

    第二步:导入api模块,添加axios的全局配置

    将api模块导入到SPA项目的src目录下,其中api模块包含了action.js(针对后台请求接口的封装定义)和http.js(针对axios的全局配置)两个文件。

    • action.js

    1. /**
    2. * 对后台请求的地址的封装,URL格式如下:
    3. * 模块名_实体名_操作
    4. */
    5. export default {
    6. 'SERVER': 'http://localhost:8080', //服务器
    7. 'SYSTEM_USER_DOLOGIN': '/user/userLogin', //登陆
    8. 'SYSTEM_USER_DOREG': '/user/userRegister', //注册
    9. 'getFullPath': k => { //获得请求的完整地址,用于mockjs测试时使用
    10. return this.SERVER + this[k];
    11. }
    12. }
    • http.js

    1. /**
    2. * vue项目对axios的全局配置
    3. */
    4. import axios from 'axios'
    5. import qs from 'qs'
    6. //引入action模块,并添加至axios的类属性urls上
    7. import action from '@/api/action'
    8. axios.urls = action
    9. // axios默认配置
    10. axios.defaults.timeout = 10000; // 超时时间
    11. // axios.defaults.baseURL = 'http://localhost:8080/j2ee15'; // 默认地址
    12. axios.defaults.baseURL = action.SERVER;
    13. //整理数据
    14. // 只适用于 POST,PUT,PATCH,transformRequest` 允许在向服务器发送前,修改请求数据
    15. axios.defaults.transformRequest = function(data) {
    16. data = qs.stringify(data);
    17. return data;
    18. };
    19. // 请求拦截器
    20. axios.interceptors.request.use(function(config) {
    21. return config;
    22. }, function(error) {
    23. return Promise.reject(error);
    24. });
    25. // 响应拦截器
    26. axios.interceptors.response.use(function(response) {
    27. return response;
    28. }, function(error) {
    29. return Promise.reject(error);
    30. });
    31. // // 路由请求拦截
    32. // // http request 拦截器
    33. // axios.interceptors.request.use(
    34. // config => {
    35. // //config.data = JSON.stringify(config.data);
    36. // //config.headers['Content-Type'] = 'application/json;charset=UTF-8';
    37. // //config.headers['Token'] = 'abcxyz';
    38. // //判断是否存在ticket,如果存在的话,则每个http header都加上ticket
    39. // // if (cookie.get("token")) {
    40. // // //用户每次操作,都将cookie设置成2小时
    41. // // cookie.set("token", cookie.get("token"), 1 / 12)
    42. // // cookie.set("name", cookie.get("name"), 1 / 12)
    43. // // config.headers.token = cookie.get("token");
    44. // // config.headers.name = cookie.get("name");
    45. // // }
    46. // return config;
    47. // },
    48. // error => {
    49. // return Promise.reject(error.response);
    50. // });
    51. // // 路由响应拦截
    52. // // http response 拦截器
    53. // axios.interceptors.response.use(
    54. // response => {
    55. // if (response.data.resultCode == "404") {
    56. // console.log("response.data.resultCode是404")
    57. // // 返回 错误代码-1 清除ticket信息并跳转到登录页面
    58. // // cookie.del("ticket")
    59. // // window.location.href='http://login.com'
    60. // return
    61. // } else {
    62. // return response;
    63. // }
    64. // },
    65. // error => {
    66. // return Promise.reject(error.response) // 返回接口返回的错误信息
    67. // });
    68. export default axios;

    第三步:修改main.js配置vue-axios

    在main.js文件中引入api模块和vue-axios模块

    1. import axios from '@/api/http'
    2. import VueAxios from 'vue-axios'
    3. Vue.use(VueAxios,axios)

    3.5 登录注册后端代码

    service层

    1. List> queryUserPager(UserVo userVo, PageBean pageBean);
    2. int insertSelective(UserVo userVo);
    1. @Override
    2. public List> queryUserPager(UserVo userVo, PageBean pageBean) {
    3. return userMapper.queryUserPager(userVo);
    4. }
    5. @Override
    6. public int insertSelective(UserVo userVo) {
    7. return userMapper.insertSelective(userVo);
    8. }

    controller层

    1. @RequestMapping("/userLogin")
    2. @ResponseBody
    3. public JsonResponseBody userLogin(UserVo userVo, HttpServletResponse response){
    4. if(userVo.getUsername().equals("admin")&&userVo.getPassword().equals("123")){
    5. //私有要求claim
    6. // Map json=new HashMap();
    7. // json.put("username", userVo.getUsername());
    8. //生成JWT,并设置到response响应头中
    9. // String jwt=JwtUtils.createJwt(json, JwtUtils.JWT_WEB_TTL);
    10. // response.setHeader(JwtUtils.JWT_HEADER_KEY, jwt);
    11. return new JsonResponseBody<>("用户登陆成功!",true,0,null);
    12. }else{
    13. return new JsonResponseBody<>("用户名或密码错误!",false,0,null);
    14. }
    15. }
    16. @RequestMapping("/userRegister")
    17. @ResponseBody
    18. public JsonResponseBody userRegistered(UserVo userVo, HttpServletRequest request){
    19. int insertSelective = userService.insertSelective(userVo);
    20. if(insertSelective>0){
    21. return new JsonResponseBody<>("用户注册成功!",true,0,null);
    22. }else{
    23. return new JsonResponseBody<>("注册失败,请稍后!",false,0,null);
    24. }
    25. }

    3.6 效果展示

    三. 跨域问题

    3.1 什么是跨域

    在浏览器同源策略限制下,向不同源(不同协议、不同域名或者不同端口)发送XHR请求,浏览器认为该请求不受信任,禁止请求,具体表现为请求后不正常响应。

    3.2 怎么解决跨域问题

    针对于本篇SPA项目,在后台代码添加一个跨域过滤器

    1. corsFilter
    2. com.zking.ssm.util.CorsFilter2
    3. corsFilter
    4. /*

  • 相关阅读:
    废水除氟技术,高盐废水除氟有什么好的方法?
    除法求值00
    嵌入式分享合集24
    docker容器内部文件挂载主机
    RT-Thread Studio学习(十二)W25Q128(SPI)的读写
    Arduino IDE + Esp32 Cam + 实现视频流 + 开发环境部署
    【docker】学习笔记
    基于ssm的班级事务管理系统设计与实现-计算机毕业设计源码+LW文档
    Python数据挖掘 | 升级版自动查核酸
    Vue2 分页
  • 原文地址:https://blog.csdn.net/lijie1025/article/details/133181134