在node环境使用MySQL

news2025/1/17 21:39:20

什么是Sequelize?

Sequelize是一个基于Promise的NodeJS ORM模块

什么是ORM?

ORM(Object-Relational-Mapping)是对象关系映射
对象关系映射可以把JS中的类和对象,和数据库中的表和数据进行关系映射。映射之后我们就可以直接通过类和对象来操作数据表和数据了, 就不用编写SQL语句了,ORM有效的解决了直接在NodeJS中编写SQL不够直观, 不够高效, 容易出错等问题。

如何映射?

  • 在Sequelize中JS中的一个类(一个模型)就对应数据库中的一张表
  • 在Sequelize中JS中的一个对象就对应表中的一条数据(一条记录)
  • 在Sequelize中JS中的一个对象的属性就对应一条数据的一个字段
|---------------------------|
|  id  |   name   |   age   |
|   1  |    zs    |    18   |
|   2  |    ls    |    19   |
|---------------------------|
// 1.创建一张表
cosnt 模型名称 = Sequelize.define('表名', {
    id: int,
    name: varchar(255),
    age: int
})
// 2.创建一条记录
let zs = 模型名称.build({
    id: 1,
    name: zs,
    age: 18
})

// 3.操作表和数据
只要是通过Sequelize定义的模型(), 那么Sequelize就会自动给这个模型添加很多操作表和数据的方法,以后我们就可以直接通过模型操作表, 通过模型创建出来的对象操作数据。

sequelize连接数据库

// 1.导入Sequelize
const Sequelize = require('sequelize');
// 2.配置连接信息
/*
第一个参数: 要操作的数据库名称
第二个参数: 数据库用户名
第三个参数: 数据库密码
第四个参数: 其它的配置信息
* */
const sequelize = new Sequelize('demo', 'root', 'root', {
    host: '127.0.0.1', // MySQL服务器地址
    port: 3306, // MySQL服务器端口号
    // 注意点: Sequelize不仅仅能操作MySQL还能够操作其它类型的数据库
    dialect: 'mysql', // 告诉Sequelize当前要操作的数据库类型
    pool: {
        max: 5, // 最多有多少个连接
        min: 0, // 最少有多少个连接
        idle: 10000, // 当前连接多久没有操作就断开
        acquire: 30000, // 多久没有获取到连接就断开
    }
});

// 3.测试配置是否正确
sequelize.authenticate()
.then(()=>{
    console.log('ok');
})
.catch((err)=>{
    console.log(err);
});

什么是数据库连接池?

默认情况下有一个人要使用数据库,那么就必须创建一个连接,默认情况下有一个人不用数据库了, 为了不占用资源, 那么就必须销毁一个连接。但是频繁的创建和销毁连接是非常消耗服务器性能的, 所以为了提升服务器性能就有了连接池。

数据库连接池是负责分配、管理和释放数据库连接,它允许应用程序重复使用一个现有的数据库连接,而不是再重新建立一个。
在这里插入图片描述

sequelize创建表

  1. 字段说明常用属性
  • type: 字段类型
  • primaryKey: 是否是主键
  • autoIncrement: 是否自动增长
  • allowNull: 是否允许为空
  • unique: 是否必须唯一
  • defaultValue: 默认值
  1. 额外配置常用属性
  • timestamps: 是否自动添加createdAt/updateAt字段
  • freezeTableName: 是否禁止自动将表名修改为复用
  • tableName: 是否自定义表名,一般不用
  • indexes:
    [ // 指定索引
        {
    	        name: '', // 索引名称
    	        fields: [''], // 索引字段名称
        }
    ]
    

实例:

/*
sequelize.define();
第一个参数: 用于指定表的名称
第二个参数: 用于指定表中有哪些字段
第三个参数: 用于配置表的一些额外信息
* */
/*
注意点:
1.sequelize在根据模型创建表的时候, 会自动将我们指定的表的名称变成复数
2.sequelize在根据模型创建表的时候, 会自动增加两个字段 createAt/updateAt
* */
let User = sequelize.define('user', {
    id: {
        type: Sequelize.INTEGER,
        primaryKey: true,
        autoIncrement: true
    },
    name: {
        type: Sequelize.STRING, // varchar(255)
        allowNull: false,
        unique: true
    },
    age: {
        type: Sequelize.TINYINT,
        defaultValue: 66
    },
    gender: {
        type: Sequelize.ENUM(['男', '女', '妖']),
        defaultValue: '妖'
    }
}, {
    freezeTableName: true, // 告诉sequelize不需要自动将表名变成复数
    // tableName: 'student', // 自定义表名,一般不用
    timestamps: false, // 不需要自动创建createAt/updateAt这两个字段
    indexes: [ // 指定索引
        {
            name: 'idx_age', // 索引名称
            fields: ['age'], // 索引字段名称
        }
    ]
});

// 注意点: 默认定义好一个模型之后并不会自动创建对应的表,我们需要通过调用连接对象的sync方法来执行同步,只有同步之后才会自动根据模型创建对应的表
sequelize.sync();

crud

// 下面这段代码 = 创建对象 + save();
 let u = await User.create({
     name: 'ww',
     age: 18,
     gender: '女'
 });
 console.log(u.dataValues);

在sequelize中,模型就代表着表,从表中查询数据,即从模型中查询数据:

 let u = await User.findByPk(2);
 console.log(u);
User.update({
   name: 'zs'
},{
    where: {
        id: 2
    }
})
User.destroy({
    where: {
        id: 3
    }
});
条件查询
let us = await User.findAll({
  where: {
       // id: 1
       // age: {
       //     [Sequelize.Op.gte] : 33
       // },
       // id: 7
       // ------上面逗号分隔是并且关系,下面是或者关系
       [Sequelize.Op.or]: {
           age: 33,
           id: 7
       }
   }
});
console.log(us.map(u => u.dataValues));
分页查询和排序
// 分页
let us = await User.findAll({
     offset: 2 // 跳过多少条数据
 });
 console.log(us.map(u => u.dataValues));

 let us = await User.findAll({
     limit: 2 // 取多少条数据
 });
 console.log(us.map(u => u.dataValues));

 let us = await User.findAll({
     offset: 2, // 跳过多少条数据
     limit: 2 // 取多少条数据
 });
 console.log(us.map(u => u.dataValues));

 // 排序
 let us = await User.findAll({
     order: [
         // ['id', 'desc']
         ['age', 'desc'],
         ['id', 'desc']
     ]
 });
 console.log(us.map(u => u.dataValues));

Sequelize关联查询:

一对一

注意点:

  • 只要建立了人和书的关系, 那么在查询人的时候, 就可以把拥有的那本书也查询出来
  • 只要建立了书和人的关系, 那么在查询书的时候, 就可以把书属于哪个人也查询出来
  • 如果没有建立相关的关系, 那么就不能查询出相关的内容
User.hasOne(Book, { // hasOne 谁拥有一个谁/ 一个人拥有一本书
     foreignKey: 'uId',
     sourceKey: 'id'
 });
 Book.belongsTo(User, { // belongsTo 谁属于一个谁 / 一本书属于一个人
     foreignKey: 'uId',
     sourceKey: 'id'
 });
let u = await User.findOne({
   where: {
       id: 1
   },
   // 注意点: 只要建立了表与表之间的关系, 那么在查询人的时候,就可以把这个人拥有的那本书也查询出来了
   include: {
       model: Book
   }
});
console.log(u.dataValues.book.dataValues);

let b = await Book.findOne({
   where: {
       id: 1
   },
   // 注意点: 只要建立了表与表之间的关系, 那么在查询书的时候,就可以把这本书属于哪一个人也查询出来了
   include: {
       model: User
   }
});
console.log(b.dataValues.user.dataValues);
一对多
User.hasMany(Book, { // 一个人拥有多本书
    foreignKey: 'uId',
    sourceKey: 'id'
});
Book.belongsTo(User, { // 一本书属于一个人
    foreignKey: 'uId',
    sourceKey: 'id'
})
// 关联查询
let u = await User.findOne({
    where: {
        id: 1
    },
    include: {
        model: Book
    }
});
console.log(u.dataValues.books.map(b=>b.dataValues));
 
let b = await Book.findOne({
    where: {
        id: 3
    },
    include: {
        model: User
    }
});
console.log(b.dataValues.user.dataValues);
多对多
 // 创建模型
let Student = sequelize.define('student', {
    id: {
        type: Sequelize.INTEGER,
        primaryKey: true,
        autoIncrement: true
    },
    name: {
        type: Sequelize.STRING, // varchar(255)
        allowNull: false,
        unique: true
    }
}, {
    freezeTableName: true, // 告诉sequelize不需要自动将表名变成复数
    // tableName: 'student', // 自定义表名
    timestamps: false // 不需要自动创建createAt/updateAt这两个字段
});

let Teacher = sequelize.define('teacher', {
    id: {
        type: Sequelize.INTEGER,
        primaryKey: true,
        autoIncrement: true
    },
    name: {
        type: Sequelize.STRING, // varchar(255)
        allowNull: false,
        unique: true
    }
}, {
    freezeTableName: true, // 告诉sequelize不需要自动将表名变成复数
    // tableName: 'student', // 自定义表名
    timestamps: false // 不需要自动创建createAt/updateAt这两个字段
});

let Relation = sequelize.define('relation', {
    studentId: {
        type: Sequelize.INTEGER,
        allowNull: false,
        references: {
            model: Student,
            key: 'id'
        }
    },
    teacherId: {
        type: Sequelize.INTEGER,
        allowNull: false,
        references: {
            model: Student,
            key: 'id'
        }
    }
}, {
    freezeTableName: true, // 告诉sequelize不需要自动将表名变成复数
    // tableName: 'student', // 自定义表名
    timestamps: false // 不需要自动创建createAt/updateAt这两个字段
});
// 建立查询关系
Student.belongsToMany(Teacher, { // 一个学生属于多个老师
    through: Relation
});
Teacher.belongsToMany(Student, { // 一个老师属于多个学生
    through: Relation
});

sequelize.sync();
// 关联查询
/*
let s = await Student.findOne({
    where: {
        id: 1
    },
    include: {
        model: Teacher
    }
});
console.log(s.dataValues.teachers.map(t=>t.dataValues));
 */
let t = await Teacher.findOne({
    where: {
        id: 1
    },
    include: {
        model: Student
    }
});
console.log(t);

Sequelize-Cli

什么是Sequelize-CLI?
  • 在编程开发中为了能够更好的管理代码, 我们可以使用Git来管理我们的代码,
    实现对代码变更的追踪, 实现在各个不同版本之间切换
  • 在数据库开发中为了能够更好的管理数据库, 我们也可以使用数据库迁移工具来管理我们的数据库,
    实现对数据库变更的追踪, 实现在各个不同版本之间切换
  • Sequelize-CLI就是一款数据库迁移工具, 能够让我们追踪数据库的变更, 在各个不同版本之间随意切换

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

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

相关文章

昇思25天学习打卡营第13天|MindNLP ChatGLM-6B StreamChat

学AI还能赢奖品?每天30分钟,25天打通AI任督二脉 (qq.com) MindNLP ChatGLM-6B StreamChat 本案例基于MindNLP和ChatGLM-6B实现一个聊天应用。 1 环境配置 %%capture captured_output # 实验环境已经预装了mindspore2.2.14,如需更换mindspo…

len()函数——计算字符串长度或元素个数

自学python如何成为大佬(目录):https://blog.csdn.net/weixin_67859959/article/details/139049996?spm1001.2014.3001.5501 语法参考 len()函数的主要功能是获取一个(字符、列表、元组等)可迭代对象的长度或项目个数。其语法格式如下: l…

精密空气加热器负载组

小型便携式 :精密空气加热器(负载组)能够对数据中心热通道/冷通道冷却系统进行全面测试。EAK 是一款 19 英寸机架式设备(10U 高),可轻松安装到各种标准服务器机架中。通过集成可调节的热量水平(…

【Android面试八股文】性能优化相关面试题: 什么是内存抖动?什么是内存泄漏?

文章目录 一、什么是内存抖动?内存抖动的问题卡顿OOM(Out Of Memory)二、什么是内存泄漏(Memory Leak)?引用计数法可达性分析法一、什么是内存抖动? 在Java中,每创建一个对象,就会申请一块内存,存储对象信息; 每分配一块内存,程序的可用内存也就少一块; 当程序…

java设计模式(十二)享元模式(Flyweight Pattern)

1、模式介绍: 享元模式是一种结构型设计模式,旨在通过共享对象来有效支持大量细粒度的对象。它通过将对象的状态分为内部状态(可共享)和外部状态(不可共享)来减少内存消耗和提高性能。内部状态存储在享元对…

SpringMVC基础详解

文章目录 一、SpringMVC简介1、什么是MVC2、MVC架构模式与三层模型的区别3、什么是SpringMVC 二、HelloWorld程序1、pom文件2、springmvc.xml3、配置web.xml文件4、html文件5、执行Controller 三、RequestMapping注解1、value属性1.1、基础使用1.2、Ant风格(模糊匹配…

详细分析Java中@RequiredArgsConstructor注解的基本知识(附Demo)

目录 前言1. 基本知识2. 源码解读3. Demo3.1 简易Demo3.2 staticName属性3.3 onConstructor属性3.4 access属性 4. AllArgsConstructor比较 前言 从源码中学习,事因是看到项目代码中有所引用 RequiredArgsConstructor 是 Lombok 提供的一个注解,用于自…

容器:deque

以下是对于deque容器知识的整理 1、构造 2、赋值 3、大小操作 4、插入 5、删除 6、数据存取 7、排序 #include <iostream> #include <deque> #include <algorithm> using namespace std; /* deque容器&#xff1a;双端数组&#xff0c;可以对头端进行插入删…

2024年06月CCF-GESP编程能力等级认证Scratch图形化编程四级真题解析

本文收录于《Scratch等级认证CCF-GESP图形化真题解析》专栏,专栏总目录:点这里,订阅后可阅读专栏内所有文章。 一、单选题(共 10 题,每题 2 分,共 30 分) 第1题 小杨父母带他到某培训机构给他报名参加 CCF 组织的 GESP 认证考试的第 1 级,那他可以选择的认证语言有几…

树状数组——点修区查与区修点查

树状数组是一种代码量小&#xff0c;维护区间的数据结构 他可以实现&#xff1a; 1.区间修改&#xff0c;单点查询 2.单点修改&#xff0c;区间查询 当然&#xff0c;二者不可兼得&#xff0c;大人全都要的话&#xff0c;请选择线段树 前置知识&#xff1a; lowbit(x)操作…

zerotier-one自建根服务器方法四

一、简介 前面几篇文章已经写完了安装配置服务器&#xff0c;今天写一下客户端如何连接自建的服务器。 二、准备工作 准备一个有公网IP的云主机。 要稳定性、安全性、不差钱的可以使用阿里、腾讯等大厂的云服务器。 本人穷屌丝一枚&#xff0c;所以我用的是免费的“三丰云…

常见sql语句练习

Tips&#xff1a;之前查看网上的文章感觉太乱了&#xff0c;所以自己整理了一套sql语句来练习&#xff0c;主要也可以拿来应对面试&#xff0c;需要的可以自行下载练习 包含基本语句、聚合函数、模糊查询、范围查询、排序、聚合、分组、分页、子查询、索引和视图、左右连接、双…

商城积分系统的代码实现(下)-- 积分订单的退款与结算

一、接着上文 用户在消耗积分的时候&#xff0c;需要根据一定的逻辑&#xff0c;除了扣减账户的当前余额&#xff0c;还需要依次消费积分订单的余额。 private void updatePointsOrderByUse(Integer schoolId, Long userId, String pointsType, int usingPoints) {List<Po…

数字证书与PKI解析

目录 1. 什么是数字证书 2. 为什么需要数字证书 3. 数字证书的格式 4. 什么是PKI 5. PKI的组成要素组件 5.1 用户 5.2 认证机构&#xff08;CA&#xff09; 5.3 仓库 5.4 PKI的体系结构 5.4.1 层次结构模型 5.4.2 交叉证明模型 5.4.3 混合模型 1. 什么是数字证书 要…

Django任意URL跳转漏洞(CVE-2018-14574)

目录 Django介绍 URL跳转漏洞介绍 Django任意URL跳转漏洞介绍 环境搭建 防御方法 前段时间在面试时&#xff0c;问到了URL跳转漏洞&#xff0c;我没有回答好&#xff0c;下午把URL跳转漏洞学习了&#xff0c;发现也不难&#xff0c;看来还需要学习的东西很多呀&#xff0c…

burp靶场xss漏洞(中级篇)下

靶场地址 All labs | Web Security Academy 第九关&#xff1a;反射型&#xff08; 转义&#xff09; 1.在搜索框随机输入字符并用Burp抓包 2.测试不同字符在JavaScript字符串中的反映&#xff0c;发现查询结果被包裹在script标签中 而单引号会被转义为 \ 3.构造payload跳出j…

Qt开发报错:Q_INTERFACES Error: Undefined interface

1、背景 VS2019qt5.12.10 从svn拉下来的项目&#xff0c;结果报错&#xff1a; Q_INTERFACES Error: Undefined interface 之前在VS的扩展中在线安装了qt插件&#xff0c; 安装了一半&#xff0c;比较慢&#xff0c;直接强行退出了。。 后来安装了qt官网的插件。。。。 2、报…

OpenCV 调用自定义训练的 YOLO-V8 Onnx 模型

一、YOLO-V8 转 Onnx 在本专栏的前面几篇文章中&#xff0c;我们使用 ultralytics 公司开源发布的 YOLO-V8 模型&#xff0c;分别 Fine-Tuning 实验了 目标检测、关键点检测、分类 任务&#xff0c;实验后发现效果都非常的不错&#xff0c;但是前面的演示都是基于 ultralytics…

【SpringCloud】Zuul源码解析

Zuul是一个网关组件&#xff0c;是微服务的入口&#xff0c;网关会根据配置将请求转发给指定的服务。本章分析Zuul组件是如何实现请求过滤和转发的 参考源码&#xff1a;<spring-cloud.version>Hoxton.SR9</spring-cloud.version> 1、过滤 spring-cloud-netflix-…

重温react-13(嵌套路由和重定向等)

重定向和404 import React from react; import { Routes, Route, Link,NavLink ,Navigate} from react-router-dom; import Home from ./Home/Home import About from ./About/About import News from ./News/News import NotFound from ./NotFound/NotFound; export default …