• 基于Vue+ELement搭建登陆注册页面实现后端交互


    🎉🎉欢迎来到我的CSDN主页!🎉🎉

    🏅我是Java方文山,一个在CSDN分享笔记的博主。📚📚

    🌟推荐给大家我的专栏《ELement》。🎯🎯

    👉点击这里,就可以查看我的主页啦!👇👇

    Java方文山的个人主页

    🎁如果感觉还不错的话请给我点赞吧!🎁🎁

    💖期待你的加入,一起学习,一起进步!💖💖

    请添加图片描述

    目录

    一、前言

    1.1.什么是ELementUI

    二、完成登陆注册前端页面

    1.1.环境搭建

    ①首先我们准备一个SPA项目

    ②运行项目

    ③使用命令添加Element-UI模块

    ④测试ELementUI

    1.2. 登录页面搭建

    ①定义组件

    ②将路由与路由路径绑定 

    ③前后端交互

    axios之get

    axios之post

     1.3.优化代码

    1.4.注册页面搭建

    ①注册页面编写

    ②后端业务处理

    三、跨域问题

    3.1.什么是跨域问题

    3.2.怎么解决跨域问题 


    一、前言

    1.1.什么是ELementUI

    Element 是一款基于 Vue.js 的开源 UI 组件库,旨在帮助开发者构建出现代化的 Web 应用程序界面。它提供了丰富多样的可重用组件,并支持自定义主题和样式,使开发者能够快速搭建出具有良好用户体验的界面。

    Element 是一个功能强大的前端 UI 组件库,通过提供丰富的组件和灵活的定制选项,使开发者能够快速构建现代化的 Web 应用程序界面。

    二、完成登陆注册前端页面

    1.1.环境搭建

    ①首先我们准备一个SPA项目

    如果不会的话,可以参照我之前写的博客

    【Vue】vue-cli一站式搭建SPA项目icon-default.png?t=N7T8http://t.csdn.cn/LlH8v

    ②运行项目

    在SAP项目的根目录输入cmd后在cmd窗口输入npm run dev

     

    ③使用命令添加Element-UI模块

    npm install element-ui -S

     先把项目终止,再进行下载,中途不可断开。

    下载完成后项目里就会看该依赖

    小贴士:指令后的-s/-g/-d有什么区别?

    1. 重视项目的稳定性和可维护性:使用-s选项安装包可以确保项目的依赖项被明确地列出。这意味着其他人在克隆项目或部署项目时能够准确地获得相同的依赖项,从而提高项目的稳定性和可维护性。

    2. 关注全局安装的适用性:某些包提供了全局命令工具,通过-g选项安装可以让这些工具在终端中以全局命令的方式使用。这对于开发人员来说很方便,可以在任意目录下使用这些命令,提高开发效率。

    3. 开发依赖项的管理:-d选项用于保存开发依赖项,这些依赖项通常是在构建、测试和调试等过程中使用的工具和库。在开发过程中,我们可能会使用一些临时性的依赖项,而不希望它们被包含在最终的生产代码中。将这些依赖项保存为devDependencies可以使项目更加整洁和可管理。

    总结:

    -g将依赖下载到node_global全局依赖;

    -d下载依赖到SPA工程中,不会参与打包;

    -s下载依赖到SPA工程中,会参与打包。

    ④测试ELementUI

    导入ELement的css与js到main.js中

    1. // 新添加1
    2. import ElementUI from 'element-ui'
    3. // 新添加2,避免后期打包样式不同,要放在import App from './App';之前
    4. import 'element-ui/lib/theme-chalk/index.css'

    注意:

    需要将CSS与JS加到APP ,因为APP是Vue中的顶级容器,之后我们还需要在APP中加入许多的ELement文件,如果ELement的css与js也在其中,页面必然会加载不出来,所以需要在这些文件之前加载好ELement的css与js。

    ELement挂载Vue

    1. // 新添加3
    2. Vue.use(ElementUI)

    总体就是这个样子

     ELementUI官网icon-default.png?t=N7T8https://element.eleme.cn/#/zh-CN

    去到ELementUI官网随便找个样式复制到SPA项目中,进行运行查看是否能运行。

    如果可以运行并且显示我们的ELementUI的按钮说明我们的环境就搭建好了

    1.2. 登录页面搭建

    ①定义组件

    首先在src下新建一个目录views,随后建立Login.Vue编写我们的页面内容以及css样式

    1. <script>
    2. export default {
    3. name: 'Login',
    4. data () {
    5. return {
    6. username:"",
    7. password:""
    8. }
    9. }
    10. }
    11. script>
    12. <style scoped>
    13. .login-wrap {
    14. box-sizing: border-box;
    15. width: 100%;
    16. height: 100%;
    17. padding-top: 10%;
    18. background-image: url();
    19. /* background-color: #112346; */
    20. background-repeat: no-repeat;
    21. background-position: center right;
    22. background-size: 100%;
    23. }
    24. .login-container {
    25. border-radius: 10px;
    26. margin: 0px auto;
    27. width: 350px;
    28. padding: 30px 35px 15px 35px;
    29. background: #fff;
    30. border: 1px solid #eaeaea;
    31. text-align: left;
    32. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    33. }
    34. .title {
    35. margin: 0px auto 40px auto;
    36. text-align: center;
    37. color: #505458;
    38. }
    39. style>

    ②将路由与路由路径绑定 

    index.js

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

    ③前后端交互

    这里我后台的代码已经写好了导入即可用,感兴趣的同志可以私信我哦!

    我们要和后台做交互就要发送ajax,我们前端不可能写$.ajax的方法去发送请求,所以需要借助依赖

    还是和以前一样先终止我们的SPA项目再进行下载。 

    1. npm i axios -S
    2. npm install qs -S

    axios 是一个用于发送 HTTP 请求和处理响应的库,方便与服务器进行数据交互。

    qs 是一个用于处理 URL 查询字符串的库,可将对象转换为查询字符串,或将查询字符串解析为对象。

    下载好后就可进行我们下面的操作 

    axios之get

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

    现把axios的依赖导入

    import axios from 'axios'

     编写登录提交的事件

    1. doSubmit(){
    2. let url="http://localhost:8080/ssm/user/userLogin";
    3. let params={
    4. username:this.username,
    5. password:this.password
    6. }
    7. //前端发送ajax的方式
    8. axios.get(url,{params:params}).then(r=>{
    9. }).catch(e=>{
    10. //如果有问题就会走这里
    11. })
    12. }

    我们先同时启动IDEA和SPA项目查看到底有没有传值过去?

    前端后端打印结果如下:

     

    没有问题,那我们继续编写成功或者失败的提示。

    1. axios.get(url, {
    2. params: params
    3. }).then(r=> {
    4. // console.log(r);
    5. if (r.data.success) {
    6. this.$message({
    7. message: r.data.msg,
    8. type: 'success'
    9. });
    10. } else {
    11. this.$message.error(r.data.msg);
    12. }
    13. }).catch(r=> {
    14. //如果有问题就会走这里
    15. // console.log(error);
    16. });

    效果展示: 

    axios之post

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

    发送 GET 请求和 POST 请求是两种常见的 HTTP 请求方法,它们有以下区别:

    1. GET 请求:GET 请求是用于获取数据的请求。当发送 GET 请求时,参数会附加在 URL 的末尾,以查询字符串的形式发送给服务器。GET 请求通常用于获取服务器上的资源,例如获取网页、获取数据列表等。GET 请求对请求参数的长度有限制,且请求参数会被包含在 URL 中,因此不适合发送敏感信息。

    2. POST 请求:POST 请求是用于提交数据的请求。当发送 POST 请求时,请求参数不会附加在 URL 上,而是作为请求体的一部分发送给服务器。POST 请求通常用于向服务器提交数据,例如表单提交、上传文件等。POST 请求对请求参数的长度没有限制,并且请求参数不会直接暴露在 URL 中,因此更适合发送较大量或敏感信息。

    总结来说,GET 请求用于获取数据,参数附加在 URL 上;而 POST 请求用于提交数据,参数作为请求体发送给服务器。一般来说,如果只是获取数据,应该使用 GET 请求;而如果需要提交数据、修改服务器状态或发送较大量的数据,应该使用 POST 请求。

    现把ps的依赖导入

    import qs from 'qs'

    将我们刚刚的get请求代码改为下面的代码即可

    1. axios.post(url,qs.stringify(params)).then(r=> {
    2. // console.log(r);
    3. if (r.data.success) {
    4. this.$message({
    5. message: r.data.msg,
    6. type: 'success'
    7. });
    8. } else {
    9. this.$message.error(r.data.msg);
    10. }
    11. }).catch(r=> {
    12. //如果有问题就会走这里
    13. // console.log(error);
    14. });

     1.3.优化代码

    刚刚上面的代码看起来是不是没有问题?但是我的URL是写死的,而且我们发送请求需要axios和qs依赖每个组件里面都需要进行导入也是十分的复制,针对这个问题做一个优化。

    首先在src下新建一个目录叫做api,随后建立action.js和http.js.

    action.js

    1. /**
    2. * 对后台请求的地址的封装,URL格式如下:
    3. * 模块名_实体名_操作
    4. */
    5. export default {
    6. 'SERVER': 'http://localhost:8080/j2ee15', //服务器
    7. 'SYSTEM_USER_DOLOGIN': '/userAction.action', //登陆
    8. 'SYSTEM_USER_DOREG': '/userAction.action', //注册
    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;

    这个文件就是用于保存公共的代码就像刚刚我们的axios和qs依赖。

    加入这两个文件我们就可以进行登录的案例吗?也不完全是,因为我们的文件还没有和Vue有任何的关系,所以我们还有借助一个依赖vue-axios来整合Vue与这两个文件。

    npm i vue-axios -S

    下载完成后也和ELement一样在main.js导入依赖

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

    这时候的Login.Vue文件也要发生改变

    1. doSubmit() {
    2. let url = this.axios.urls.SYSTEM_USER_DOLOGIN;
    3. let params = {
    4. username: this.username,
    5. password: this.password
    6. }
    7. //前端发送ajax的post方式
    8. this.axios.post(url,params).then(r=> {
    9. // console.log(r);
    10. if (r.data.success) {
    11. this.$message({
    12. message: r.data.msg,
    13. type: 'success'
    14. });
    15. } else {
    16. this.$message.error(r.data.msg);
    17. }
    18. }).catch(r=> {
    19. //如果有问题就会走这里
    20. // console.log(error);
    21. });
    22. }

    1.4.注册页面搭建

    ①注册页面编写

    复制Login.Vue改名为Registered.Vue并配置路由与路由绑定

    回到我们的Login.Vue编写@click="gotoRegister()的方法

    这里提一个东西,Vue里面的router-link相当于a标签,而location相当于this.$router.push

     Login.Vue

    1. methods: {
    2. gotoRegister() {
    3. this.$router.push('/Registered')
    4. }
    5. }

     Registered.Vue 

    1. <script>
    2. export default {
    3. name: 'Registered',
    4. data() {
    5. return {
    6. username: "",
    7. password: ""
    8. }
    9. },
    10. methods: {
    11. gotoLogin() {
    12. this.$router.push('/')
    13. }
    14. }
    15. }
    16. script>
    17. <style scoped>
    18. .login-wrap {
    19. box-sizing: border-box;
    20. width: 100%;
    21. height: 100%;
    22. padding-top: 10%;
    23. background-image: url();
    24. /* background-color: #112346; */
    25. background-repeat: no-repeat;
    26. background-position: center right;
    27. background-size: 100%;
    28. }
    29. .login-container {
    30. border-radius: 10px;
    31. margin: 0px auto;
    32. width: 350px;
    33. padding: 30px 35px 15px 35px;
    34. background: #fff;
    35. border: 1px solid #eaeaea;
    36. text-align: left;
    37. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    38. }
    39. .title {
    40. margin: 0px auto 40px auto;
    41. text-align: center;
    42. color: #505458;
    43. }
    44. style>

    我们先看一下效果 

    ②后端业务处理

    首先我们注册肯定不能填写全部的字段,只需要账户和密码即可所以我们选择方法是选择性插入,只插入非空属性的。

    IUserService

    1. int insertSelective(User record);

    实现IUserService

    1. @Override
    2. public int insertSelective(User record) {
    3. return userMapper.insertSelective(record);
    4. }

    实现IUserService

    1. @RequestMapping("/userRegistered")
    2. @ResponseBody
    3. public JsonResponseBody userRegistered(UserVo userVo, HttpServletRequest request){
    4. int i = userService.insertSelective(userVo);
    5. if(i>0){
    6. return new JsonResponseBody<>("用户注册成功!",true,0,null);
    7. }else{
    8. return new JsonResponseBody<>("注册失败,请稍后!",false,0,null);
    9. }
    10. }

    Registered.Vue 

    1. <script>
    2. export default {
    3. name: 'Registered',
    4. data() {
    5. return {
    6. username: "",
    7. password: ""
    8. }
    9. },
    10. methods: {
    11. gotoLogin() {
    12. this.$router.push('/')
    13. },
    14. doSubmit() {
    15. let url = this.axios.urls.SYSTEM_USER_DOREG;
    16. let params = {
    17. username: this.username,
    18. password: this.password
    19. }
    20. this.axios.post(url,params).then(r=> {
    21. // console.log(r);
    22. if (r.data.success) {
    23. this.$message({
    24. message: r.data.msg,
    25. type: 'success',
    26. });
    27. this.$router.push('/')
    28. } else {
    29. this.$message.error(r.data.msg);
    30. }
    31. }).catch(r=> {
    32. //如果有问题就会走这里
    33. // console.log(error);
    34. });
    35. }
    36. }
    37. }
    38. script>
    39. <style scoped>
    40. .login-wrap {
    41. box-sizing: border-box;
    42. width: 100%;
    43. height: 100%;
    44. padding-top: 10%;
    45. background-image: url();
    46. /* background-color: #112346; */
    47. background-repeat: no-repeat;
    48. background-position: center right;
    49. background-size: 100%;
    50. }
    51. .login-container {
    52. border-radius: 10px;
    53. margin: 0px auto;
    54. width: 350px;
    55. padding: 30px 35px 15px 35px;
    56. background: #fff;
    57. border: 1px solid #eaeaea;
    58. text-align: left;
    59. box-shadow: 0 0 20px 2px rgba(0, 0, 0, 0.1);
    60. }
    61. .title {
    62. margin: 0px auto 40px auto;
    63. text-align: center;
    64. color: #505458;
    65. }
    66. style>

    注意配置URL

    'SYSTEM_USER_DOREG': '/user/userRegistered', //注册

     效果展示:

    我们看一下数据库有没有该数据

    三、跨域问题

    3.1.什么是跨域问题

    跨域指的是在浏览器中,当前网页的域名与发送请求的目标资源所在的域名不一致。具体来说,当浏览器发起一个跨域请求时,会受到同源策略(Same Origin Policy)的限制。

    同源策略是浏览器的一种安全机制,它限制了来自不同源(域名、协议、端口)的 JavaScript 代码对当前文档的访问。换句话说,JavaScript 只能访问与其所在文档具有相同源的资源,而无法直接访问跨域的资源。

    跨域请求的常见场景包括:

    1. 域名不同:例如从 www.example.com 发起请求到 api.example.com
    2. 协议不同:例如从 http://example.com 发起请求到 https://example.com
    3. 端口不同:例如从 example.com:3000 发起请求到 example.com:8000

    跨域请求可以引发安全问题,因此浏览器会对这类请求进行限制。为了在跨域情况下实现数据交互,可以使用一些方法,如 JSONP、CORS(跨域资源共享)、代理等来解决跨域问题。

    以上就是跨域问题出现的提示信息,但我们看见这个错误就要知道是跨域问题。

    3.2.怎么解决跨域问题 

    那为什么前面没有这个问题,因为我在我的Maven项目里已经做了这个处理

    CorsFilter2.java

    1. package com.zking.ssm.util;
    2. import java.io.IOException;
    3. import javax.servlet.Filter;
    4. import javax.servlet.FilterChain;
    5. import javax.servlet.FilterConfig;
    6. import javax.servlet.ServletException;
    7. import javax.servlet.ServletRequest;
    8. import javax.servlet.ServletResponse;
    9. import javax.servlet.http.HttpServletResponse;
    10. /**
    11. * 配置tomcat允许跨域访问
    12. *
    13. * @author Administrator
    14. *
    15. */
    16. public class CorsFilter2 implements Filter {
    17. @Override
    18. public void init(FilterConfig filterConfig) throws ServletException {
    19. }
    20. @Override
    21. public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain)
    22. throws IOException, ServletException {
    23. HttpServletResponse httpResponse = (HttpServletResponse) servletResponse;
    24. // Access-Control-Allow-Origin就是我们需要设置的域名
    25. // Access-Control-Allow-Headers跨域允许包含的头。
    26. // Access-Control-Allow-Methods是允许的请求方式
    27. httpResponse.setHeader("Access-Control-Allow-Origin", "*");// *,任何域名
    28. httpResponse.setHeader("Access-Control-Allow-Headers", "responseType,Origin, X-Requested-With, Content-Type, Accept");
    29. httpResponse.setHeader("Access-Control-Allow-Methods", "POST, GET, PUT, DELETE");
    30. //允许客户端处理一个新的响应头jwt
    31. //httpResponse.setHeader("Access-Control-Expose-Headers", "jwt,Content-Disposition");
    32. filterChain.doFilter(servletRequest, servletResponse);
    33. }
    34. @Override
    35. public void destroy() {
    36. }
    37. }

    web.xml

    1. <filter>
    2. <filter-name>corsFilterfilter-name>
    3. <filter-class>com.zking.ssm.util.CorsFilter2filter-class>
    4. filter>
    5. <filter-mapping>
    6. <filter-name>corsFilterfilter-name>
    7. <url-pattern>/*url-pattern>
    8. filter-mapping>

    因为我自定义的过滤器,用于在 Java Web 应用中配置 Tomcat 允许跨域访问。它设置了适当的 CORS 响应头,以允许跨域请求,包括允许的来源域、请求头和请求方法。这样可以解决浏览器跨域访问时的安全限制,确保请求能够正常发送并获取响应。

    请添加图片描述

    到这里我的分享就结束了,欢迎到评论区探讨交流!!

    💖如果觉得有用的话还请点个赞吧 💖

  • 相关阅读:
    jmeter分布式压力测试搭建
    Swift中的Error处理
    基于深度学习的Wiki中文语料词word2vec向量模型
    transforms.Normalize()
    双链表的插入和删除
    苹果如何做ASO优化?优化的主要思路有哪些?
    使用U3D、pico开发VR(一)——将unity的场景在设备中呈现
    项目绩效评估七宗罪
    【Java集合类面试十八】、ConcurrentHashMap是怎么分段分组的?
    国际版阿里云/腾讯云免费开户:什么是DDos攻击?
  • 原文地址:https://blog.csdn.net/weixin_74318097/article/details/133180079