添加登录和注册功能

先写前端再写后端

前提:idea+vue3+mybatis+springBoot3前后端分离实现对一张表的增删改查(完整代码版)-CSDN博客

项目地址

1.添加一个Login.vue视图

<template><div class="login_container"><div class="login_box"><div style="padding:20px;background-color: white;margin-left: 100px;border-radius: 20px;box-shadow:0 0  10px rgba(0,0,0.1)"><el-form ref="formRef" :rules="data.rules" :model="data.form" style="width:400px"  autocomplete="off" ><div style="margin-bottom:30px;font-size:20px;color:#0742b1;font-weight:bolder;text-align: center">欢迎登录后台管理系统</div><el-form-item style="margin-top: 20px;" prop="username" autocomplete="off"><el-input size="large"   v-model="data.form.username" placeholder="请输入账号" prefix-icon="User" ></el-input></el-form-item><el-form-item  prop="password"><el-input show-password="show-password" autocomplete="off" size="large" v-model="data.form.password" placeholder="请输入密码" prefix-icon="Lock"></el-input></el-form-item><div><el-button size="large" style="width: 48%" type="primary" @click="login()">登 录</el-button><el-button size="large" style="width: 48%" type="info">取 消</el-button></div><div style="text-align: right;margin-top: 20px;">还没有账号?请<a style="color: #0742b1;text-decoration: none;" href="/register">注册</a></div></el-form></div></div></div>
</template>
<script setup>
import { reactive,ref} from "vue";
import request from "@/utils/request.js";
import {ElMessage} from "element-plus";const res = ref()
const data = reactive({form:{username:'',password:''},rules:{username:[{required:true,message:'请输入账号',trigger:'blur'}],password:[{required:true,message:'请输入密码',trigger:'blur'}]}
})
const formRef = ref()
const login = () => {formRef.value.validate((valid) => {if (valid) {request.post('/employee/login', data.form).then(res => {if (res.code === '200') {localStorage.setItem("zwy-user",JSON.stringify(res.data))//把json对象转换为字符串ElMessage.success('登录成功');location.href = '/manager/home'} else {ElMessage.error(res.msg); // 显示业务错误消息}}).catch(error => {if (error.response) {// 处理 HTTP 错误状态码(如 400)const { data } = error.response;if (data && data.msg) {ElMessage.error(data.msg); // 显示后端返回的错误消息} else {ElMessage.error('请求失败,请稍后重试');}} else {ElMessage.error('网络请求失败,请检查网络连接');}});}});
}</script>
<!--scoped表示当页生效-->
<style scoped>
.login_container{height: 100vh;overflow:hidden;background-image: url("@/assets/login_bg.png");background-size:cover ;background-position: 0 -40px;
}
.login_box{width: 50%;height: 100%;display: flex;align-items: center;right: 0;position: absolute;
}
</style>

背景图为下图 位置background-image: url("@/assets/login_bg.png");

2.添加一个Register.vue视图

<template><div class="Register_container"><div class="Register_box"><div style="padding:20px;background-color: white;margin-left: 100px;border-radius: 20px;box-shadow:0 0  10px rgba(0,0,0.1)"><el-form ref="formRef" :rules="data.rules" :model="data.form" style="width:400px" ><div style="margin-bottom:30px;font-size:20px;color:#0742b1;font-weight:bolder;text-align: center">欢迎注册后台管理系统</div><el-form-item style="margin-top: 20px;" prop="username"><el-input size="large" autocomplete="off" v-model="data.form.username" placeholder="请输入账号" prefix-icon="User" ></el-input></el-form-item><el-form-item  prop="password"><el-input show-password="show-password" autocomplete="off" size="large" v-model="data.form.password" placeholder="请输入密码" prefix-icon="Lock"></el-input></el-form-item><el-form-item  prop="confirmPassword"><el-input show-password="show-password" autocomplete="off" size="large" v-model="data.form.confirmPassword" placeholder="请再次输入密码" prefix-icon="Lock"></el-input>{{data.form.confirmPassword}}</el-form-item><div><el-button size="large" style="width: 48%" type="primary" @click="register()">注 册</el-button><el-button size="large" style="width: 48%" type="info">取 消</el-button></div><div style="text-align: right;margin-top: 20px;">已有账号?请<a style="color: #0742b1;text-decoration: none;" href="/login">登录</a></div></el-form></div></div></div>
</template>
<script setup>
import { reactive,ref} from "vue";
import request from "@/utils/request.js";
import {ElMessage} from "element-plus";const res = ref()const validatePass = (rule, value, callback) => {if (!value) {callback(new Error('请再次输入密码'))} else if (value !== data.form.password) {callback(new Error("两次输入密码不一致!"))} else {callback()}
}
const data = reactive({form:{username:'',password:'',confirmPassword:''},rules:{username:[{required:true,message:'请输入账号',trigger:'blur'}],password:[{required:true,message:'请输入密码',trigger:'blur'}],confirmPassword:[{validator:validatePass,trigger:'blur'}]}
})
const formRef = ref()
const register = () => {formRef.value.validate((valid) => {if (valid) {request.post('/register', data.form).then(res => {if (res.code === '200') {ElMessage.success('注册成功');location.href = '/login'} else {ElMessage.error(res.msg); // 显示业务错误消息}}).catch(error => {if (error.response) {// 处理 HTTP 错误状态码(如 400)const { data } = error.response;if (data && data.msg) {ElMessage.error(data.msg); // 显示后端返回的错误消息} else {ElMessage.error('请求失败,请稍后重试');}} else {ElMessage.error('网络请求失败,请检查网络连接');}});}});
}</script>
<!--scoped表示当页生效-->
<style scoped>
.Register_container{height: 100vh;overflow:hidden;background-image: url("@/assets/login_bg.png");background-size:cover ;background-position: 0 -40px;
}
.Register_box{width: 50%;height: 100%;display: flex;align-items: center;right: 0;position: absolute;
}
</style>

3.添加两个视图的路由

下面是index.js的完整代码

import { createRouter, createWebHistory } from 'vue-router'const router = createRouter({history: createWebHistory(import.meta.env.BASE_URL),routes: [{path: '/',redirect: '/login',},{path: '/button',name: 'button',component: () => import('../views/Button.vue'),},{path:'/manager',name:'manager',component:()=>import('../views/Manager.vue'),children:[{path:'/manager/employeeDataOp',name:'employeeDataOp',meta:{title:'员工数据操作'},component:()=>import('../views/employeeDataOp.vue')},{path:'/manager/home',name:'home',meta:{title:'首页'},component:()=>import('../views/home.vue')},{path:'/manager/adminDataOp',name:'adminDataOp',meta:{title:'管理员数据操作'},component:()=>import('../views/adminDataOp.vue')}]},{path:'/login',name:'login',meta:{title:'登录界面'},component:()=>import('../views/Login.vue')},{path:'/form',name:'form',component:()=>import('../views/form.vue')},{path:'/register',name:'register',meta:{title:'注册界面'},component:()=>import('../views/Register.vue')},{path:'/404',name:'404',meta:{title:'404'},component:()=>import('../views/404.vue')},{path:'/:pathMatch(.*)*',redirect:'/404',}],
})
router.beforeEach((to,from,next)=>{document.title=to.meta.title;next()
})export default router

4.后端添加两个功能的API接口

EmployeeController.java文件

    @PostMapping("/login")public Result login(@RequestBody Employee employee){Employee findEmployee = employeeService.login(employee);return Result.success(findEmployee);}@PostMapping("/register")public Result register(@RequestBody Employee employee){Employee employee1 = employeeService.register(employee);return Result.success(employee1);}

5.为接口添加业务层方法

  public Employee login(Employee employee) {Employee findEmployee = employeeMapper.selectByUsername(employee.getUsername());if (findEmployee == null){throw new ServiceException(ErrorCode.EMPLOYEE_NOT_FOUND);}if (!findEmployee.getPassword().equals(employee.getPassword())){throw new ServiceException(ErrorCode.EMPLOYEE_LOGIN_ERROR);}return findEmployee;}public Employee register(Employee employee){Employee findEmployee = employeeMapper.selectByUsername(employee.getUsername());employee.setRole("EMP");employee.setName("默认名称");employee.setGender("男");employee.setTitle("普通员工");employee.setBirthday(new Date());if (findEmployee != null){throw new ServiceException(ErrorCode.EMPLOYEE_REGISTER_ERROR_USERNAME);}int row =  employeeMapper.insert(employee);if(row!=1){throw new ServiceException(ErrorCode.EMPLOYEE_ADD_ERROR);}return employee;}

添加依赖

    <dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.8.37</version></dependency>

对这里将原先的EmployeeServcieException.java改为ServcieException.java

然后ErrorCode加几个常量下面是完整代码

package com.cdp.zwy.zwy_manager_back_v1.common;/*** @description: 业务异常枚举类* @author bug制作者* @date 2025/4/6 20:40* @version 1.0*/public enum ErrorCode {NOT_DATA("10000","没有数据"),EMPLOYEE_NOT_FOUND("10001", "该员工不存在"),EMPLOYEE_ADD_ERROR("10002", "添加员工失败"),EMPLOYEE_DELETE_ERROR("10003", "删除员工失败"),EMPLOYEE_MODIFY_ERROR("10004", "修改员工失败"),EMPLOYEE_LOGIN_ERROR("10005", "登录失败"),EMPLOYEE_REGISTER_ERROR_USERNAME("10006","不能注册,该用户已经存在");private final String code;private final String msg;ErrorCode(String code, String msg) {this.code = code;this.msg = msg;}public String getCode() {return code;}public String getMsg() {return msg;}}

测试接口

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/76888.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Windows】系统安全移除移动存储设备指南:告别「设备被占用」弹窗

Windows系统安全移除移动存储设备指南&#xff1a;告别「设备被占用」弹窗 解决移动硬盘和U盘正在被占用无法弹出 一、问题背景 使用Windows系统时&#xff0c;经常遇到移动硬盘/U盘弹出失败提示「设备正在使用中」&#xff0c;即使已关闭所有可见程序。本文将系统梳理已验证…

Springboot下载文件, 文件名中文是乱码, 空格变加号

默认把文件名放上去, 中文会乱码, 文件名种有空格, 就会被截断 public void download(HttpServletResponse response){// 文件名先进行url编码, 避免乱码问题// 把用%20进行替换fileName URLEncoder.encode(fileName, "UTF-8").replace("", "%20&qu…

MySQL 超详细安装教程与常见问题解决方案

一、MySQL 安装教程 1. Windows 系统安装&#xff08;以 MySQL 8.0 为例&#xff09; 步骤 1&#xff1a;下载 MySQL Installer 访问 MySQL 官网下载页面。 选择 Windows (x86, 64-bit), MSI Installer&#xff08;推荐使用完整版 mysql-installer-web-community-8.0.xx.xx.…

【cuda学习日记】5.2.1 共享内存额外篇

共享内存(Shared Memory) 1.是一种低延迟、高带宽的片上内存 2.由同一个Block内的所有线程共享 3.生命周期与Block相同 4.访问速度比全局内存快约100倍 Block(线程块) 1.GPU执行的基本单位&#xff0c;包含一组线程 2.多个Block组成Grid(网格) 3.Block内的线程可以通过共享内存…

[250411] Meta 发布 Llama 4 系列 AI 模型 | Rust 1.86 引入重大语言特性

目录 Llama 4 家族登场&#xff1a;开启原生多模态 AI 创新新纪元Rust 1.86.0 版本发布亮点主要新特性与改进其他重要信息 Llama 4 家族登场&#xff1a;开启原生多模态 AI 创新新纪元 Meta AI 近日发布了其最新、最先进的 Llama 4 系列人工智能模型&#xff0c;标志着 AI 技术…

ArrayList 和 数组 的区别

定义与本质 数组&#xff1a;是 Java 语言内置的数据结构&#xff0c;是存储相同类型元素的连续内存空间。它是一个基本的语言特性&#xff0c;在内存中是一块连续的区域。ArrayList&#xff1a;是 Java 集合框架中的一个类&#xff0c;属于动态数组。它是基于数组实现的&#…

​‌FireCrawl‌爬虫工具​, Craw4ai

‌FireCrawl‌是一款开源的AI爬虫工具&#xff0c;专门用于Web数据提取&#xff0c;并将其转换为Markdown格式或其他结构化数据。FireCrawl特别适合处理使用JavaScript动态生成的网站&#xff0c;能够自动抓取网站及其所有可访问的子页面内容&#xff0c;并将其转换为适合大语言…

通信原理-非线性调制

今天给大家带来的是关于通信原理中非线性调制的内容,一起来看看吧&#xff01;&#xff01;&#xff01; 1.角度调制 2.FM与PM的区别 3.单音调制FM 4.窄带调频 5.宽带调频 5.1FM信号的频谱 5.2FM信号的带宽 5.3FM信号的功率分配 6.FM信号的产生与解调 6.1FM信号的产生 6.2FM…

文心一言开发指南03——千帆大模型平台产品优势

版权声明 本文原创作者&#xff1a;谷哥的小弟作者博客地址&#xff1a;http://blog.csdn.net/lfdfhl 千帆大模型平台作为百度智能云推出的企业级大模型一站式平台&#xff0c;具有显著的产品优势。千帆大模型平台以其基础强大、流程完善、运行稳定和安全可靠的产品优势成为企…

mysql DQL

一.基本查询 1.查询多个字段 2.查看所有字段 3.设置别名 4.去除重复记录 二.条件查询 1.大于小于等于 2.查询 身份证为空的 没有所以没有记录 3.在15到20这个区间范围内 4.or/in 或者 4.like 匹配 &#xff08;_匹配单个字符 %匹配多个字符&#xff09; 查询员工信…

关于 软件开发模型 的分类、核心特点及详细对比分析,涵盖传统模型、迭代模型、敏捷模型等主流类型

以下是关于 软件开发模型 的分类、核心特点及详细对比分析&#xff0c;涵盖传统模型、迭代模型、敏捷模型等主流类型&#xff1a; 一、软件开发模型分类及核心特点 1. 瀑布模型&#xff08;Waterfall Model&#xff09; 核心特点&#xff1a; 线性阶段划分&#xff1a;需求分…

2025年第十六届蓝桥杯省赛C++ A组真题

2025年第十六届蓝桥杯省赛C A组真题 1.说明2.题目A&#xff1a;寻找质数&#xff08;5分&#xff09;3.题目B&#xff1a;黑白棋&#xff08;5分&#xff09;4. 题目C&#xff1a;抽奖&#xff08;10分&#xff09;5. 题目D&#xff1a;红黑树&#xff08;10分&#xff09;6. 题…

JVM初探——走进类加载机制|三大特性 | 打破双亲委派SPI机制详解

目录 JVM是什么&#xff1f; 类加载机制 Class装载到JVM的过程 装载&#xff08;load&#xff09;——查找和导入class文件 链接&#xff08;link&#xff09;——验证、准备、解析 验证&#xff08;verify&#xff09;——保证加载类的正确性 准备&#xff08;Prepare&…

分布式微服务系统架构第106集:jt808,补充类加载器

加群联系作者vx&#xff1a;xiaoda0423 仓库地址&#xff1a;https://webvueblog.github.io/JavaPlusDoc/ https://1024bat.cn/ 类加载器 类与类加载器 判断类是否“相等” 任意一个类&#xff0c;都由加载它的类加载器和这个类本身一同确立其在 Java 虚拟机中的唯一性&#xf…

利用 pyecharts 实现地图的数据可视化——第七次人口普查数据的2d、3d展示(关键词:2d 、3d 、map、 geo、涟漪点)

参考文档&#xff1a;链接: link_pyecharts 官方文档 1、map() 传入省份全称&#xff0c;date_pair 是列表套列表 [ [ ],[ ] … ] 2、geo() 传入省份简称&#xff0c;date_pair 是列表套元组 [ ( ),( ) … ] 1、准备数据 population_data&#xff1a;简称经纬度 population_da…

Enovia许可释放

随着企业规模的扩大和业务的不断增长&#xff0c;Enovia许可证的管理变得至关重要。在许多情况下&#xff0c;企业可能面临许可证资源浪费或不足的问题。为了解决这一问题&#xff0c;Enovia提供了许可释放功能&#xff0c;帮助企业更加灵活地管理和使用许可证资源。本文将介绍…

每日一道leetcode(回来了!!!)

236. 二叉树的最近公共祖先 - 力扣&#xff08;LeetCode&#xff09; 题目 给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。 百度百科中最近公共祖先的定义为&#xff1a;“对于有根树 T 的两个节点 p、q&#xff0c;最近公共祖先表示为一个节点 x&#xff0c;满足…

【Redis】布隆过滤器应对缓存穿透的go调用实现

布隆过滤器 https://pkg.go.dev/github.com/bits-and-blooms/bloom/v3 作用&#xff1a; 判断一个元素是不是在集合中 工作原理&#xff1a; 一个位数组&#xff08;bit array&#xff09;&#xff0c;初始全为0。多个哈希函数&#xff0c;运算输入&#xff0c;从而映射到位数…

【ROS2】行为树 BehaviorTree(四):组合使用子树

1、大树调用子树 如下图,左边为大树主干: 1)如果门没有关,直接通过; 2)如果门关闭了,执行开门动作,然后通过 右边为子树,主要任务是开门 1)尝试直接开门; 2)尝试开锁开门,最多尝试5次; 3)最后尝试砸门! XML如何描述大树主干调佣子树:使用关键字 SubTree 来…