express实现用户登录和注册接口

目录

  • 1 创建数据库
  • 2 连接数据库
  • 3 集成ORM库
  • 4 创建业务逻辑
  • 5 创建路由
  • 7 测试接口
  • 总结

我们在编写后端接口的时候操作数据库是一种常见的功能需求,express本身并不提供直接操作数据库的能力,需要借助第三方库来操作数据库,本篇讲解一下软件开发中像登录和注册如何实现。

1 创建数据库

通常我们编制软件的时候需要选择数据库,我这里选择mysql作为示例,如果你有比较喜欢的数据库也可以按照相同的操作进行。

安装数据库可以参考网上的教程,因为比较普遍这里就不赘述了。数据库安装好之后,用数据库的连接工具来操作数据,我这里用的是navicate,你也可以用别的。

打开我们的navicate,点击连接
在这里插入图片描述
选择mysql
在这里插入图片描述
输入连接名,因为我数据库是在本地,直接输入localhost就行,然后输入数据库的用户名和密码,点击测试连接
在这里插入图片描述
然后点击你的连接名,右键,点击新建数据库
在这里插入图片描述
输入数据库的名称,字符集选择utf8mb4,排序规则选择第一项就可以
在这里插入图片描述
双击数据库名称,选择表,点击新建表
在这里插入图片描述
创建四个字段,分别是username、password、email、id,前三个选择varchar类型就可以,第四个字段需要作为主键,勾选自动增长
在这里插入图片描述
然后点击保存按钮,输入表名即可
在这里插入图片描述
除了可视化的建表外,我们还可以使用sql语句来建表,以下是建表语句

CREATE TABLE users (  id INT AUTO_INCREMENT PRIMARY KEY,  username VARCHAR(50) NOT NULL UNIQUE,  password VARCHAR(255) NOT NULL,  email VARCHAR(100) NOT NULL UNIQUE,  createdAt TIMESTAMP DEFAULT CURRENT_TIMESTAMP,  updatedAt TIMESTAMP DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP  
);

还是在navicate里,点击查询、新建查询,输入我们的建表语句
在这里插入图片描述
在这里插入图片描述
点击运行,看到输出的OK就表示表已经创建成功了

2 连接数据库

连接数据库通常需要配置数据库的IP、用户名和密码,我们创建一个环境变量的配置文件来存储这些信息,打开我们的工程,在根目录下创建一个.env的文件
在这里插入图片描述
贴入如下配置

DATABASE_NAME=express  
DATABASE_USER=root  
DATABASE_PASSWORD=111111  
DATABASE_HOST=localhost

具体的配置信息要改成你自己设定的数据库的信息

然后在app.js中读取我们的配置

require('dotenv').config();//加载环境变量

我们使用了dotenv的库,需要先进行安装

npm install dotenv

在这里插入图片描述
完成上述步骤你就可以看到我们的配置信息被正确的读取到了

3 集成ORM库

我们这里使用sequelize来做数据库的操作,orm的好处可以屏蔽具体的数据库语句的细节,在操作数据库的时候可以使用他封装好的API,省了不少事。而且后期如果你想切换数据库,改一下配置文件就可以了。

在项目的根目录创建一个config文件夹,里边创建一个database.js
在这里插入图片描述
贴入如下的配置信息

// 导入 Sequelize 和相应的数据库驱动  
const { Sequelize } = require('sequelize');  
const sequelize = new Sequelize(process.env.DATABASE_NAME, process.env.DATABASE_USER, process.env.DATABASE_PASSWORD, {  host: 'localhost',  dialect: 'mysql', // 根据你的数据库类型选择 'mysql' | 'mariadb' | 'postgres' | 'mssql' | 'sqlite' | 'postgres-native' | 'mariadb-native'  // 其他可能的配置选项  dialectOptions: {  useUTC: true, // 对于不支持 UTC 的数据库,启用这个选项  // 针对特定数据库的额外配置,比如 SSL  },  pool: {  max: 5, // 连接池中的最大连接数  min: 0, // 连接池中的最小连接数  acquire: 30000, // 连接最大等待时间(毫秒)  idle: 10000 // 连接在释放之前可以空闲的最长时间(毫秒)  },  // 其他的 Sequelize 配置项...  logging: console.log, // 启用日志,你可以选择自己的日志函数或者禁用它  
});  // 测试连接  
sequelize  .authenticate()  .then(() => {  console.log('Connection has been established successfully.');  })  .catch(err => {  console.error('Unable to connect to the database:', err);  });  // 导出 Sequelize 实例供其他模块使用  
module.exports = sequelize;

添加配置信息后我们先需要按照库,执行如下命令

npm install sequelize
npm install mysql2

配置创建好之后,需要创建模型层,新建一个model文件夹,里边创建一个User.js
在这里插入图片描述
输入如下配置

// models/User.js  
const { Sequelize, Model } = require('sequelize');  
const sequelize = require('../config/database'); // 假设你有一个数据库配置文件  class User extends Model {}  User.init(  {  email: {  type: Sequelize.STRING,  unique: true,  allowNull: false,  },  password: {  type: Sequelize.STRING,  allowNull: false,  }, username: {type:Sequelize.STRING,allowNull: false, } // 其他字段...  },  {  sequelize,  modelName: 'User',  }  
);  module.exports = User;

如果你的模型和表名不一致,需要指定表名,我这里使用了默认的推断机制,他会转换为小写加一个s的后缀去匹配表,正好是我们的users表

4 创建业务逻辑

我们把业务逻辑封装到service层,这样代码的维护性要好一点,在根目录创建services文件夹,创建一个authService.js

// services/authService.js  
const bcrypt = require('bcrypt');  
const jwt = require('jsonwebtoken');  
const User = require('../models/User');  class AuthService {  async signUp(username, password,email) {  console.log(username,password,email)const salt = await bcrypt.genSalt(10);  const hashedPassword = await bcrypt.hash(password, salt);  const user = await User.create({ username,email, password: hashedPassword });  return user;  }  async signIn(username, password) {  const user = await User.findOne({ where: { username } });  if (!user) throw new Error('User not found');  const isMatch = await bcrypt.compare(password, user.password); console.log("isMatch",isMatch) if (!isMatch) throw new Error('Invalid credentials');  const token = jwt.sign({ userId: user.id }, process.env.JWT_SECRET_KEY, { expiresIn: '1h' });  return { user, token };  }  
}  module.exports = AuthService;

这里我们写了两个方法,一个是注册,注册的时候需要给密码加密,登录的时候需要返回给用户一个token,需要安装需要的第三方的库

npm install bcrypt jsonwebtoken

因为我们在创建token时候需要一个密钥,我们也在配置信息里添加,在.env里增加密钥

JWT_SECRET_KEY=d6b522d9a989a9ff8494676b4e1592ea58488b8a4f8126d33d331604e769f4d3

5 创建路由

一切准备就绪之后最后一步就是创建路由了,我们在根目录创建routes,然后创建auth.js
在这里插入图片描述
输入如下代码

// routes/auth.js  
const express = require('express');  
const router = express.Router();  
const AuthService = require('../services/authService');  
const authService = new AuthService();  
// 登录路由  
router.post('/signIn', async (req, res, next) => {  try {  const { username, password } = req.body;  console.log(username,password)console.log(authService)const { user, token } = await authService.signIn(username, password);  res.cookie('token', token, { httpOnly: true, secure: true }); // 假设使用cookie存储token  return res.status(200).json({ code:200,data:{token:token},message:'login success' });  } catch (error) { console.log('error',error.message) if (error.message === 'Invalid credentials'||error.message ==='User not found') {  // 如果是因为无效的凭证,返回401状态码  res.status(401).json({ code:401,message: 'Invalid credentials' });  } else {  // 其他错误,返回500状态码  res.status(500).json({ code:500,message: 'Internal server error' });  // 可能还需要记录错误日志  }  //next(error);  }  
});  // 注册路由  
router.post('/signUp', async (req, res, next) => {  try {  const { username, password ,email} = req.body;  const user = await authService.signUp(username,password,email);  res.status(200).json({code:200,id:user.id});  } catch (error) {  //next(error);res.status(500).json({ code:500,message: 'Internal server error' });   }  
});  module.exports = router;

我们的路由创建好之后要挂载到应用程序里,修改app.js增加路由的挂载代码

const authRouter = require('./routes/auth')//引入权限接口路由
app.use('/api/auth', authRouter);//挂载权限路由

因为接口涉及到参数解析,我们还要启用解析的中间件

// 对于 JSON 请求体  
app.use(express.json());  // 对于 URL 编码的请求体  
app.use(express.urlencoded({ extended: true }));

7 测试接口

接口开发完毕后,就可以进行测试了,这里使用Postman来测试,先测试一下注册接口
在这里插入图片描述
然后再测试一下登录接口
在这里插入图片描述
可以看到两个接口都已经正常返回数据,这样就可以了

总结

本篇我们讲解了一下用户登录和注册接口的开发,涉及到环境变量的初始化,Orm库的搭建以及路由的配置,熟悉了整套流程,后续的功能开发就可以按照自己的需求完成了。

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

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

相关文章

【二叉树】Leetcode 543. 二叉树的直径【简单】

二叉树的直径 给你一棵二叉树的根节点,返回该树的 直径 。 二叉树的 直径 是指树中任意两个节点之间最长路径的 长度 。这条路径可能经过也可能不经过根节点 root 。 两节点之间路径的 长度 由它们之间边数表示。 示例1: 输入:root [1,2…

基于SpringBoot + Vue实现的在线装修管理系统设计与实现+毕业论文

介绍 系统包含用户、装修队、管理员三个角色 管理员: 管理员管理:管理其他管理员的账号和权限,确保系统管理的层次化和安全性。 装修队管理:审核装修队的资质,管理装修队的人员信息,监控工程进度&#xff…

elementUI this.$msgbox msgBox自定义 样式自定义 富文本

看这个效果是不是很炫?突出重点提示内容,对于用户交互相当的棒! 下来说说具体实现: let self = this const h = self.$createElement; this.$msgbox({title: null,message: h("p", {style: "margin-top:10px"}, [h("i", {class: "el-i…

命名空间【C++】(超详细)

文章目录 命名空间的概念命名空间的定义命名空间定义的位置作用域每一个命名空间都是一个独立的域作用域符:: 编译器找一个变量/函数等的定义,寻找域的顺序为什么要有命名空间?1.解决库与程序员定义的同名的重定义问题2.解决程序员…

【氮化镓】p-GaN栅极退化的温度和结构相关性

论文总结: 本文献深入研究了带有p-GaN栅极的正常关断型(normally-off)高电子迁移率晶体管(GaN-HEMTs)在恒定电压应力下的时序退化行为。通过直流特性分析和温度依赖性分析,研究了故障时间(TTF)与应力温度和器件几何结构的依赖性。结果显示,p…

FME学习之旅---day17

我们付出一些成本,时间的或者其他,最终总能收获一些什么。 【FME-HOW-TO系列】28 栅格邻域函数 RasterConvolver转换器说明: 接受包含栅格几何对象的输入要素,并在对所有波段应用卷积滤波 器后输出要素。 本人对栅格数据处理的较…

【2023】kafka在linux和docker安装(kafka-1)

目录💻 一、linux安装kafka1. 安装jdk2. 上传解压到/usr/local目录下3、使用kafka 二、docker安装kafka1. 下载2. 安装zookeeper3. 安装kafka 一、linux安装kafka 环境主机 mac m2、虚拟机Ubuntu22.04.4 1. 安装jdk yum install -y java-1.8.0-openjdk.x86_64下载k…

11-设计模式:Go常用设计模式概述

设计模式是啥呢?简单来说,就是将软件开发中需要重复性解决的编码场景,按最佳实践的方式抽象成一个模型,模型描述的解决方法就是设计模式。使用设计模式,可以使代码更易于理解,保证代码的重用性和可靠性。 …

【Entity Framework】EF中DbSet类详解

【Entity Framework】EF中DbSet类详解 文章目录 【Entity Framework】EF中DbSet类详解一、概述二、定义DbSet2.1 具有DbSet属性的DbContext2.2 具有 IDbSet 属性的 DbContext 2.3 具有 IDbSet 属性的 DbContext三、DbSet属性四、DbSet方法五、DbContext动态生成DbSet 一、概述 …

打工人神器! Raccoon 代码小浣熊

继这三个之后,今天又来了一个 [ Raccoon代码小浣熊 ] 核心精要与产品特点 全面支持多种编程语言和IDE:「代码小浣熊」支持超过90种主流编程语言,包括但不限于Python、Java、JavaScript、C、Go和SQL等。同时,它集成了市面上主流的…

Quiet-STaR:让语言模型在“说话”前思考

大型语言模型(llm)已经变得越来越复杂,能够根据各种提示和问题生成人类质量的文本。但是他们的推理能力让仍然是个问题,与人类不同LLM经常在推理中涉及的隐含步骤中挣扎,这回导致输出可能在事实上不正确或缺乏逻辑。 考虑以下场景:正在阅读一…

CTF题型 php://filter特殊编码绕过小汇总

CTF题型 php://filter特殊编码绕过小汇总 文章目录 CTF题型 php://filter特殊编码绕过小汇总特殊编码base64编码string过滤器iconv字符集 例题1.[Newstarctf 2023 week2 include]2.[Ctfshow web 117] php://filter 是一个伪协议,它允许你读取经过过滤器处理的数据流…

【C++】string类(常用接口)

🌈个人主页:秦jh__https://blog.csdn.net/qinjh_?spm1010.2135.3001.5343🔥 系列专栏:http://t.csdnimg.cn/eCa5z 目录 修改操作 push_back append operator assign insert erase replace c_str find string类非成…

AI学习-Pandas数据处理分析

文章目录 1. Pandas概述2. Series用法2.1 Series的创建2.2 Series的取值2.3 Series的相关方法 3. DataFrame用法3.1 DataFrame创建3.2 DataFrame取值3.3 DataFrame相关方法 1. Pandas概述 ​ Pandas 是一个开源的数据分析处理库,它应用在数据科学、统计分析、机器学…

9.0-源码分析:Dubbo Remoting 层核心接口分析

dubbo-remoting 模块,该模块提供了多种客户端和服务端通信的功能。在 Dubbo 的整体架构设计图中,我们可以看到最底层红色框选中的部分即为 Remoting 层,其中包括了 Exchange、Transport和Serialize 三个子层次。这里我们要介绍的 dubbo-remot…

C++类继承继承5——构造函数与拷贝控制

构造函数与拷贝控制 和其他类一样,位于继承体系中的类也需要控制当其对象执行一系列操作时发生什么样的行为,这些操作包括创建、拷贝、移动、赋值和销毁。 如果一个类(基类或派生类)没有定义拷贝控制操作,则编译器将为它合成一个版本。当然…

手写简易操作系统(十七)--编写键盘驱动

前情提要 上一节我们实现了锁与信号量,这一节我们就可以实现键盘驱动了,访问键盘输入的数据也属于临界区资源,所以需要锁的存在。 一、键盘简介 之前的 ps/2 键盘使用的是中断驱动的,在当时,按下键盘就会触发中断&a…

乐理通识

2023 年搞了台雅马哈 61 键的电子琴,顺手看了下啊 B 的上的课程 《零基础自学音乐学乐理合集-第一季》,这里是部分笔记(给博客加点不一样的东西👀)。 简谱各部分一览 C 表示音名竖线为小节线 音名 完整钢琴键盘 88 键…

数据结构

一、栈 先进后出 二、队列 先进先出 三、数组 查询快,增加修改慢 四、链表 查询慢,增加修改慢 五、二叉树 节点: 查找二叉树 二叉查找树的特点 二叉查找树,又称二叉排序树或者二叉搜索树 每一个节点上最多有两个子节点 左子树上所…

Linux shell编程学习笔记43:cut命令

0 前言 在 Linux shell编程学习笔记42:md5sum 中,md5sum命令计算md5校验值后返回信息的格式是: md5校验值 文件名 包括两项内容,前一项是md5校验值 ,后一项是文件名。 如果我们只想要前面的md5 校验值&#xff0c…