“构建完善的用户认证与数据交互系统“

本文涉及的产品
.cn 域名,1个 12个月
简介: “构建完善的用户认证与数据交互系统“

引言

在现代Web应用程序开发中,用户认证和数据交互是至关重要的功能。本文将介绍如何使用ElementUI、axios和解决跨域问题来构建一个完善的用户认证与数据交互系统。我们将分别探讨ElementUI完成登录注册、axios之get请求、axios之post请求以及跨域问题的解决方案。

1.ElementUI完成登录注册

在构建一个Web应用程序时,用户登录和注册是必不可少的功能。ElementUI是一个流行的Vue.js组件库,提供了丰富的UI组件和样式,使得开发者可以轻松地构建出漂亮且易用的登录和注册页面。本节将介绍如何使用ElementUI来完成登录和注册功能,并展示一些常用的验证和表单处理技巧。

1. 登录页面设计与实现

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
import ElementUI from 'element-ui'
// 新添加2,避免后期打包样式不同,要放在import App from './App';之前
import 'element-ui/lib/theme-chalk/index.css'
import App from './App'
import router from './router'
// 新添加3
Vue.use(ElementUI)
Vue.config.productionTip = false
/* eslint-disable no-new */
import axios from '@/api/http'
import VueAxios from 'vue-axios'
Vue.use(VueAxios,axios)
new Vue({
  el: '#app',
  router,
  components: { App },
  template: '<App/>'
})
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25

// 新添加3

Vue.use(ElementUI)

Vue.config.productionTip = false

在app后面添加

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

配置路由

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

登录界面

<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>
    <script>
      // import axios from 'axios'
      // import qs from 'qs'
      export default {
        name: 'Login',
        data() {
          return {
            username: '',
            password: ''
          }
        },
        methods: {
          gotoRegister() {
            this.$router.push('/Register');
          },
          doSubmit() {
            let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
            let params = {
              username: this.username,
              password: this.password
            }
            this.axios.post(url, params).then(r => {
              console.log(r)
              if (r.data.success) {
                this.$message({
                  type: 'success',
                  message: r.data.msg
                });
                this.$router.push('/Register');
              }else{
                console.log(r)
                this.$message(r.data.msg)
              }
            }).catch(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>
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25
• 26
• 27
• 28
• 29
• 30
• 31
• 32
• 33
• 34
• 35
• 36
• 37
• 38
• 39
• 40
• 41
• 42
• 43
• 44
• 45
• 46
• 47
• 48
• 49
• 50
• 51
• 52
• 53
• 54
• 55
• 56
• 57
• 58
• 59
• 60
• 61
• 62
• 63
• 64
• 65
• 66
• 67
• 68
• 69
• 70
• 71
• 72
• 73
• 74
• 75
• 76
• 77
• 78
• 79
• 80
• 81
• 82
• 83
• 84
• 85
• 86
• 87
• 88
• 89
• 90
• 91
• 92
• 93
• 94
• 95
• 96

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>
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18

2. 注册页面设计与实现

注册页面与登录页面类似,也需要一个表单来接收用户输入的注册信息,并提供注册按钮来提交表单。在注册页面中,我们还可以使用ElementUI的其他组件,如下拉框、日期选择器等,来提供更多的注册选项。同样,我们也可以利用ElementUI的表单验证功能来确保用户输入的合法性。

<template>
  <div class="register-container">
    <div class="register-card">
      <h2 class="register-title">注册</h2>
      <form @submit.prevent="register" class="register-form">
        <div class="form-group">
          <label for="username" class="register-label">用户名:</label>
          <input type="text" id="username" v-model="username" required class="register-input">
        </div>
        <div class="form-group">
          <label for="password" class="register-label">密码:</label>
          <input type="password" id="password" v-model="password" required class="register-input">
        </div>
        <div class="form-group">
          <label for="password" class="register-label">编号:</label>
          <input type="password" id="id" v-model="id" required class="register-input">
        </div>
        <button type="submit" class="register-button">注册</button>
      </form>
      <div class="login-link">
        <span>已有账号?</span>
         <el-link type="primary" @click="gotoLogin()">用户登录</el-link>
      </div>
    </div>
  </div>
</template>
<style>
.register-container {
  display: flex;
  flex-direction: column;
  align-items: center;
  justify-content: center;
  height: 100vh;
  background-color: #f5f5f5;
}
.register-card {
  display: flex;
  flex-direction: column;
  align-items: center;
  background-color: #fff;
  padding: 40px;
  border-radius: 4px;
  box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
}
.register-title {
  font-size: 24px;
  color: #333;
  margin-bottom: 20px;
}
.register-form {
  display: flex;
  flex-direction: column;
  align-items: center;
}
.form-group {
  margin-bottom: 20px;
}
.register-label {
  font-size: 16px;
  color: #333;
  margin-bottom: 10px;
}
.register-input {
  padding: 8px;
  border: 1px solid #ccc;
  border-radius: 4px;
  width: 200px;
}
.register-button {
  padding: 10px 20px;
  background-color: #ff6700;
  color: #fff;
  border: none;
  border-radius: 4px;
  cursor: pointer;
}
.login-link {
  margin-top: 20px;
  display: flex;
  align-items: center;
}
.login-link span {
  color: #555;
}
.login-button {
  margin-left: 5px;
  color: #007bff;
  text-decoration: none;
}
</style>
<script>
export default {
  data() {
    return {
      username: '',
      password: '',
      id:''
    };
  },
  methods: {
    register() {
      let url = this.axios.urls.SYSTEM_USER_DOREG;
      let params = {
        username: this.username,
        password: this.password,
        id:this.id
      }
console.log(url)
        this.axios.post(url, params).then(r => {
          console.log(r)
          if (r.data.success) {
            this.$message({
              type: 'success',
              message: r.data.msg
            });
          }else{
            console.log(r)
            this.$message(r.data.msg)
          }
        }).catch(e => {
        })
    },
    gotoLogin(){
        this.$router.push('/');
    }
  }
};
</script>
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25
• 26
• 27
• 28
• 29
• 30
• 31
• 32
• 33
• 34
• 35
• 36
• 37
• 38
• 39
• 40
• 41
• 42
• 43
• 44
• 45
• 46
• 47
• 48
• 49
• 50
• 51
• 52
• 53
• 54
• 55
• 56
• 57
• 58
• 59
• 60
• 61
• 62
• 63
• 64
• 65
• 66
• 67
• 68
• 69
• 70
• 71
• 72
• 73
• 74
• 75
• 76
• 77
• 78
• 79
• 80
• 81
• 82
• 83
• 84
• 85
• 86
• 87
• 88
• 89
• 90
• 91
• 92
• 93
• 94
• 95
• 96
• 97
• 98
• 99
• 100
• 101
• 102
• 103
• 104
• 105
• 106
• 107
• 108
• 109
• 110
• 111
• 112
• 113
• 114
• 115
• 116
• 117
• 118
• 119
• 120
• 121
• 122
• 123
• 124
• 125
• 126
• 127
• 128
• 129
• 130
• 131
• 132
• 133
• 134
• 135
• 136
• 137
• 138
• 139
• 140
• 141
• 142
• 143

后台代码

package com.zking.ssm.controller;
import com.zking.ssm.model.User;
import com.zking.ssm.service.IUserService;
import com.zking.ssm.util.JsonResponseBody;
import com.zking.ssm.util.PageBean;
import com.zking.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.zking.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("/userRegister")
    @ResponseBody
    public JsonResponseBody<?> Register(User user, HttpServletResponse response) {
        if (user.getUsername() != null && user.getPassword() != null && user != null) {
            int n = userService.insertSelective(user);
            if (n > 0) {
                return new JsonResponseBody<>("注册成功!", true, 0, null);
            }
        }
        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);
        }
    }
}
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25
• 26
• 27
• 28
• 29
• 30
• 31
• 32
• 33
• 34
• 35
• 36
• 37
• 38
• 39
• 40
• 41
• 42
• 43
• 44
• 45
• 46
• 47
• 48
• 49
• 50
• 51
• 52
• 53
• 54
• 55
• 56
• 57
• 58
• 59
• 60
• 61
• 62
• 63
• 64
• 65
• 66
• 67
• 68
• 69
• 70
• 71
• 72
• 73

在现代Web应用程序中,与后端服务器进行数据交互是非常常见的需求。axios是一个基于Promise的HTTP客户端,可以在浏览器和Node.js中发送HTTP请求。本节将介绍如何使用axios发送GET请求,并展示一些常用的请求参数配置和响应处理技巧。

项目下输入下面代码回车下载

npm i axios -S

doSubmit() {
            let url = "http://localhost:8080/user/userLogin"
            let params = {
              username: this.username,
              password: this.password
            }
            axios.get(url, {
              params: params
            }).then(r => {
              console.log(r)
              if (r.data.success) {
                this.$message({
                  type: 'success',
                  message: r.data.msg
                });
                this.$router.push('/Register');
              }else{
                console.log(r)
                this.$message(r.data.msg)
              }
            }).catch(e => {
            })
          }
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25

3.axios之post请求

除了GET请求,POST请求也是常用的数据交互方式之一。POST请求通常用于向服务器提交数据,如用户注册、表单提交等。本节将介绍如何使用axios发送POST请求,并展示一些常用的请求参数配置和响应处理技巧。

项目下输入下面代码回车下载

npm install qs -S

npm i vue-axios -S

doSubmit() {
            let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
            let params = {
              username: this.username,
              password: this.password
            }
            this.axios.post(url, params).then(r => {
              console.log(r)
              if (r.data.success) {
                this.$message({
                  type: 'success',
                  message: r.data.msg
                });
                this.$router.push('/Register');
              }else{
                console.log(r)
                this.$message(r.data.msg)
              }
            }).catch(e => {
            })
          }
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22

action.js

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

http.js

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

4.跨域问题的解决方案

在前后端分离的开发模式中,前端代码通常运行在一个独立的域名或端口上,而后端API则运行在另一个域名或端口上。由于浏览器的同源策略限制,跨域请求会受到限制。本节将介绍一些常用的跨域解决方案,如JSONP、CORS等,并展示如何使用axios来处理跨域请求。

web.xml配置过滤器

<!--CrosFilter跨域过滤器-->
  <filter>
    <filter-name>corsFilter</filter-name>
    <filter-class>com.zking.ssm.util.CorsFilter2</filter-class>
  </filter>
  <filter-mapping>
    <filter-name>corsFilter</filter-name>
    <url-pattern>/*</url-pattern>
  </filter-mapping>
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9

过滤器

package com.zking.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.HttpServletResponse;
/**
 * 配置tomcat允许跨域访问
 * 
 * @author Administrator
 *
 */
public class CorsFilter2 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.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
    httpResponse.setHeader("Access-Control-Allow-Headers", "responseType,Origin, X-Requested-With, Content-Type, Accept");
    httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
    //允许客户端处理一个新的响应头jwt
    //httpResponse.setHeader("Access-Control-Expose-Headers", "jwt,Content-Disposition");
    filterChain.doFilter(servletRequest, servletResponse);
  }
  @Override
  public void destroy() {
  }
}
• 1
• 2
• 3
• 4
• 5
• 6
• 7
• 8
• 9
• 10
• 11
• 12
• 13
• 14
• 15
• 16
• 17
• 18
• 19
• 20
• 21
• 22
• 23
• 24
• 25
• 26
• 27
• 28
• 29
• 30
• 31
• 32
• 33
• 34
• 35
• 36
• 37
• 38
• 39
• 40
• 41
• 42
• 43
• 44
• 45
• 46

5.总结

本文介绍了如何使用ElementUI、axios和解决跨域问题来构建一个完善的用户认证与数据交互系统。我们分别探讨了ElementUI完成登录注册、axios之get请求、axios之post请求以及跨域问题的解决方案。通过学习本文,您将掌握一些高级的前端技术实践,能够更好地应对现代Web应用程序开发中的挑战。希望本文对您有所帮助,谢谢阅读!


相关文章
|
19天前
|
存储 前端开发 安全
如何确保前端框架数据驱动方式的数据加密存储的兼容性?
确保前端框架数据驱动方式的数据加密存储的兼容性需要综合考虑多个因素,通过充分的评估、测试、关注和更新,以及与其他技术的协调配合,来提高兼容性的可靠性,为用户提供稳定和安全的使用体验。
27 2
|
4月前
|
开发者 存储 API
Xamarin 云服务集成竟然如此强大,简化后端开发不再是梦,数据存储、用户认证、推送通知全搞定!
【8月更文挑战第31天】Xamarin 是一款强大的跨平台移动应用开发工具,通过与云服务集成,显著简化了后端开发。开发者无需自行搭建服务器,即可利用云服务提供的数据存储、用户认证、推送通知等功能,大幅减少数据库设计、服务器配置及 API 开发的时间成本。借助 Azure Mobile Apps 等云服务,Xamarin 可轻松实现数据存取操作,同时增强应用安全性与用户参与度,使开发者更专注于业务逻辑和用户体验,提升开发效率并降低成本。这种方式在快速发展的移动应用领域极具价值。
60 0
|
4月前
|
前端开发 开发者 UED
数据校验的艺术:揭秘JSF如何将前端与后端验证合二为一,打造无缝用户体验
【8月更文挑战第31天】JavaServer Faces(JSF)是构建企业级Web应用的Java规范,提供了丰富的组件和API,便于快速搭建用户界面。JSF验证框架基于JavaBean验证API(JSR 303/JSR 380),利用注解如`@NotNull`、`@Size`等在模型类上定义验证规则,结合前端的`&lt;h:inputText&gt;`和`&lt;h:message&gt;`标签展示错误信息。
46 0
|
4月前
|
Java Spring 开发者
掌握Spring事务管理,打造无缝数据交互——实用技巧大公开!
【8月更文挑战第31天】在企业应用开发中,确保数据一致性和完整性至关重要。Spring框架提供了强大的事务管理机制,包括`@Transactional`注解和编程式事务管理,简化了事务处理。本文深入探讨Spring事务管理的基础知识与高级技巧,涵盖隔离级别、传播行为、超时时间等设置,并介绍如何使用`TransactionTemplate`和`PlatformTransactionManager`进行编程式事务管理。通过合理设计事务范围和选择合适的隔离级别,可以显著提高应用的稳定性和性能。掌握这些技巧,有助于开发者更好地应对复杂业务需求,提升应用质量和可靠性。
49 0
|
存储 前端开发 安全
强化用户体验与安全性:前端单点登录和统一认证的最佳实践与区别
互联网发展了这么多年,各种更新皆为了提供更好更安全的上网环境。同时为了提供更好的用户体验、减少用户反复输入用户名和密码的繁琐操作,并确保账户安全,前端领域中的单点登录(SSO)和统一认证(Unified Authentication)成为了重要概念。
强化用户体验与安全性:前端单点登录和统一认证的最佳实践与区别
|
7月前
|
监控 安全 测试技术
最终用户验证:关键一环
最终用户验证:关键一环
|
7月前
|
存储 数据库 数据安全/隐私保护
用户认证过程的详细解析,保护你的数据安全
用户认证过程的详细解析,保护你的数据安全
88 3
|
安全 API 数据处理
API接口技术:架构、设计、安全性与实践
随着数字化和信息化进程的加速,应用程序接口(API)在软件开发和数据处理中扮演着越来越重要的角色。API不仅是不同软件应用程序之间的桥梁,也是软件开发人员和数据科学家进行数据交换和功能调用的关键工具。本文将深入探讨API接口的技术,包括架构、设计、安全性以及实践应用。
|
JSON 自然语言处理 前端开发
跨端架构下客户端侧API维护方案总结
淘宝App搜索业务侧采用的是局部动态化的跨端技术架构,客户端提供丰富的基础能力与视图组件的API,前端负责业务视图搭建与业务逻辑实现。
121 0
|
存储 运维 安全
如何做好统一身份认证账号管理及集成
传统统一身份认证系统的建设存在众多的问题,使设计实现复杂化,管理复杂化,集成复杂化。我们今天将详细讨论下统一身份认证账号设计的几个相关问题
2555 1
如何做好统一身份认证账号管理及集成