Sequelize 提供了多种方法来协助查询数据库中的数据.
重要说明:要使用 Sequelize 执行生产级别的查询,请确保你还阅读了事务指南. 事务对于确保数据完整性和提供其它好处很重要.
本指南将说明如何进行标准的 增删改查(CRUD) 查询.
简单 INSERT 查询#
首先,一个简单的例子:
const jane = await User.create({ firstName: "Jane", lastName: "Doe" });
console.log("Jane's auto-generated ID:", jane.id);
Model.create()
方法是使用 Model.build()
构建未保存实例并使用 instance.save()
保存实例的简写形式.
也可以定义在 create
方法中的属性. 如果你基于用户填写的表单创建数据库条目,这将特别有用. 例如,使用它可以允许你将 User
模型限制为仅设置用户名和地址,而不设置管理员标志:
const user = await User.create({
username: 'alice123',
isAdmin: true
}, { fields: ['username'] });
console.log(user.username);
console.log(user.isAdmin);
简单 SELECT 查询#
你可以使用 findAll
方法从数据库中读取整个表:
const users = await User.findAll();
console.log(users.every(user => user instanceof User));
console.log("All users:", JSON.stringify(users, null, 2));
SELECT 查询特定属性#
选择某些特定属性,可以使用 attributes
参数:
Model.findAll({
attributes: ['foo', 'bar']
});
可以使用嵌套数组来重命名属性:
Model.findAll({
attributes: ['foo', ['bar', 'baz'], 'qux']
});
SELECT foo, bar AS baz, qux FROM ...
你可以使用 sequelize.fn
进行聚合:
Model.findAll({
attributes: [
'foo',
[sequelize.fn('COUNT', sequelize.col('hats')), 'n_hats']
'bar'
]
});
SELECT foo, COUNT(hats) AS n_hats, bar FROM ...
使用聚合函数时,必须为它提供一个别名,以便能够从模型中访问它. 在上面的示例中,你可以通过 instance.n_hats
获取帽子数量.
有时,如果只想添加聚合,那么列出模型的所有属性可能会很麻烦:
Model.findAll({
attributes: [
'id', 'foo', 'bar', 'baz', 'qux', 'hats',
[sequelize.fn('COUNT', sequelize.col('hats')), 'n_hats']
]
});
Model.findAll({
attributes: {
include: [
[sequelize.fn('COUNT', sequelize.col('hats')), 'n_hats']
]
}
});
SELECT id, foo, bar, baz, qux, hats, COUNT(hats) AS n_hats FROM ...
同样,也可以排除某些属性:
Model.findAll({
attributes: { exclude: ['baz'] }
});
SELECT id, foo, bar, qux FROM ...
应用 WHERE 子句#
where
参数用于过滤查询.where
子句有很多运算符,可以从 Op
中以 Symbols 的形式使用.
Post.findAll({
where: {
authorId: 2
}
});
可以看到没有显式传递任何运算符(来自Op
),因为默认情况下 Sequelize 假定进行相等比较. 上面的代码等效于:
const { Op } = require("sequelize");
Post.findAll({
where: {
authorId: {
[Op.eq]: 2
}
}
});
可以传递多个校验:
Post.findAll({
where: {
authorId: 12
status: 'active'
}
});
就像在第一个示例中 Sequelize 推断出 Op.eq
运算符一样,在这里 Sequelize 推断出调用者希望对两个检查使用 AND
. 上面的代码等效于:
const { Op } = require("sequelize");
Post.findAll({
where: {
[Op.and]: [
{ authorId: 12 },
{ status: 'active' }
]
}
});
OR
可以通过类似的方式轻松执行:
const { Op } = require("sequelize");
Post.findAll({
where: {
[Op.or]: [
{ authorId: 12 },
{ authorId: 13 }
]
}
});
由于以上的 OR
涉及相同字段 ,因此 Sequelize 允许你使用稍有不同的结构,该结构更易读并且作用相同:
const { Op } = require("sequelize");
Post.destroy({
where: {
authorId: {
[Op.or]: [12, 13]
}
}
});
操作符#
Sequelize 提供了多种运算符.
const { Op } = require("sequelize");
Post.findAll({
where: {
[Op.and]: [{ a: 5 }, { b: 6 }],
[Op.or]: [{ a: 5 }, { b: 6 }],
someAttribute: {
[Op.eq]: 3,
[Op.ne]: 20,
[Op.is]: null,
[Op.not]: true,
[Op.or]: [5, 6],
[Op.col]: 'user.organization_id',
[Op.gt]: 6,
[Op.gte]: 6,
[Op.lt]: 10,
[Op.lte]: 10,
[Op.between]: [6, 10],
[Op.notBetween]: [11, 15],
[Op.all]: sequelize.literal('SELECT 1'),
[Op.in]: [1, 2],
[Op.notIn]: [1, 2],
[Op.like]: '%hat',
[Op.notLike]: '%hat',
[Op.startsWith]: 'hat',
[Op.endsWith]: 'hat',
[Op.substring]: 'hat',
[Op.iLike]: '%hat',
[Op.notILike]: '%hat',
[Op.regexp]: '^[h|a|t]',
[Op.notRegexp]: '^[h|a|t]',
[Op.iRegexp]: '^[h|a|t]',
[Op.notIRegexp]: '^[h|a|t]',
[Op.any]: [2, 3],
[Op.like]: { [Op.any]: ['cat', 'hat'] }
}
}
});
Op.in
的简写语法#
直接将数组参数传递给 where
将隐式使用 IN
运算符:
Post.findAll({
where: {
id: [1,2,3]
}
});
运算符的逻辑组合#
运算符 Op.and
, Op.or
和 Op.not
可用于创建任意复杂的嵌套逻辑比较.
使用 Op.and
和 Op.or
示例#
const { Op } = require("sequelize");
Foo.findAll({
where: {
rank: {
[Op.or]: {
[Op.lt]: 1000,
[Op.eq]: null
}
},
{
createdAt: {
[Op.lt]: new Date(),
[Op.gt]: new Date(new Date() - 24 * 60 * 60 * 1000)
}
},
{
[Op.or]: [
{
title: {
[Op.like]: 'Boat%'
}
},
{
description: {
[Op.like]: '%boat%'
}
}
]
}
}
});
使用 Op.not
示例#
Project.findAll({
where: {
name: 'Some Project',
[Op.not]: [
{ id: [1,2,3] },
{
description: {
[Op.like]: 'Hello%'
}
}
]
}
});
上面将生成:
SELECT *
FROM `Projects`
WHERE (
`Projects`.`name` = 'a project'
AND NOT (
`Projects`.`id` IN (1,2,3)
OR
`Projects`.`description` LIKE 'Hello%'
)
)
高级查询(不仅限于列)#
如果你想得到类似 WHERE char_length("content") = 7
的结果怎么办?
Post.findAll({
where: sequelize.where(sequelize.fn('char_length', sequelize.col('content')), 7)
});
请注意方法 sequelize.fn
和 sequelize.col
的用法,应分别用于指定 SQL 函数调用和列. 应该使用这些方法,而不是传递纯字符串(例如 char_length(content)
),因为 Sequelize 需要以不同的方式对待这种情况(例如,使用其他符号转义方法).
如果你需要更复杂的东西怎么办?
Post.findAll({
where: {
[Op.or]: [
sequelize.where(sequelize.fn('char_length', sequelize.col('content')), 7),
{
content: {
[Op.like]: 'Hello%'
}
},
{
[Op.and]: [
{ status: 'draft' },
sequelize.where(sequelize.fn('char_length', sequelize.col('content')), {
[Op.gt]: 10
})
]
}
]
}
});
上面生成了以下SQL:
SELECT
...
FROM "posts" AS "post"
WHERE (
char_length("content") = 7
OR
"post"."content" LIKE 'Hello%'
OR (
"post"."status" = 'draft'
AND
char_length("content") > 10
)
)
仅限 Postgres 的范围运算符#
可以使用所有支持的运算符查询范围类型.
请记住,提供的范围值也可以定义绑定的 包含/排除.
[Op.contains]: 2,
[Op.contains]: [1, 2],
[Op.contained]: [1, 2],
[Op.overlap]: [1, 2],
[Op.adjacent]: [1, 2],
[Op.strictLeft]: [1, 2],
[Op.strictRight]: [1, 2],
[Op.noExtendRight]: [1, 2],
[Op.noExtendLeft]: [1, 2],
不推荐使用: 操作符别名#
在 Sequelize v4 中,可以指定字符串来引用运算符,而不是使用 Symbols. 现在不建议使用此方法,很可能在下一个主要版本中将其删除. 如果确实需要,可以在 Sequelize 构造函数中传递 operatorAliases
参数.
例如:
const { Sequelize, Op } = require("sequelize");
const sequelize = new Sequelize('sqlite::memory:', {
operatorsAliases: {
$gt: Op.gt
}
});
Foo.findAll({
where: {
$gt: 6
}
});
简单 UPDATE 查询#
Update 查询也接受 where
参数,就像上面的读取查询一样.
await User.update({ lastName: "Doe" }, {
where: {
lastName: null
}
});
简单 DELETE 查询#
Delete 查询也接受 where
参数,就像上面的读取查询一样.
await User.destroy({
where: {
firstName: "Jane"
}
});
要销毁所有内容,可以使用 TRUNCATE
SQL:
await User.destroy({
truncate: true
});
批量创建#
Sequelize 提供了 Model.bulkCreate
方法,以允许仅一次查询即可一次创建多个记录.
通过接收数组对象而不是单个对象,Model.bulkCreate
的用法与 Model.create
非常相似.
const captains = await Captain.bulkCreate([
{ name: 'Jack Sparrow' },
{ name: 'Davy Jones' }
]);
console.log(captains.length);
console.log(captains[0] instanceof Captain);
console.log(captains[0].name);
console.log(captains[0].id);
但是,默认情况下,bulkCreate
不会在要创建的每个对象上运行验证(而 create
可以做到). 为了使 bulkCreate
也运行这些验证,必须通过validate: true
参数. 但这会降低性能. 用法示例:
const Foo = sequelize.define('foo', {
bar: {
type: DataTypes.TEXT,
validate: {
len: [4, 6]
}
}
});
await Foo.bulkCreate([
{ name: 'abc123' },
{ name: 'name too long' }
]);
await Foo.bulkCreate([
{ name: 'abc123' },
{ name: 'name too long' }
], { validate: true });
如果你直接从用户获取值,那么限制实际插入的列可能会有所帮助. 为了做到这一点,bulkCreate()
接受一个 fields
参数,该参数须为你要定义字段的数组(其余字段将被忽略).
await User.bulkCreate([
{ username: 'foo' },
{ username: 'bar', admin: true }
], { fields: ['username'] });
排序和分组#
Sequelize 提供了 order
and group
参数,来与 ORDER BY
和 GROUP BY
一起使用.
order
参数采用一系列 项 来让 sequelize 方法对查询进行排序. 这些 项 本身是 [column, direction]
形式的数组. 该列将被正确转义,并且将在有效方向列表中进行验证(例如 ASC
, DESC
, NULLS FIRST
等).
Subtask.findAll({
order: [
['title', 'DESC'],
sequelize.fn('max', sequelize.col('age')),
[sequelize.fn('max', sequelize.col('age')), 'DESC'],
[sequelize.fn('otherfunction', sequelize.col('col1'), 12, 'lalala'), 'DESC'],
[Task, 'createdAt', 'DESC'],
[Task, Project, 'createdAt', 'DESC'],
['Task', 'createdAt', 'DESC'],
['Task', 'Project', 'createdAt', 'DESC'],
[Subtask.associations.Task, 'createdAt', 'DESC'],
[Subtask.associations.Task, Task.associations.Project, 'createdAt', 'DESC'],
[{model: Task, as: 'Task'}, 'createdAt', 'DESC'],
[{model: Task, as: 'Task'}, {model: Project, as: 'Project'}, 'createdAt', 'DESC']
],
order: sequelize.literal('max(age) DESC'),
order: sequelize.fn('max', sequelize.col('age')),
order: sequelize.col('age'),
order: sequelize.random()
});
Foo.findOne({
order: [
['name'],
['username', 'DESC'],
sequelize.fn('max', sequelize.col('age')),
[sequelize.fn('max', sequelize.col('age')), 'DESC'],
[sequelize.fn('otherfunction', sequelize.col('col1'), 12, 'lalala'), 'DESC'],
[sequelize.fn('otherfunction', sequelize.fn('awesomefunction', sequelize.col('col'))), 'DESC']
]
});
回顾一下,order 数组的元素可以如下:
- 一个字符串 (它将被自动引用)
- 一个数组, 其第一个元素将被引用,第二个将被逐字追加
- 一个具有
raw
字段的对象:raw
内容将不加引用地逐字添加- 其他所有内容都将被忽略,如果未设置
raw
,查询将失败
- 调用
Sequelize.fn
(这将在 SQL 中生成一个函数调用) - 调用
Sequelize.col
(这将引用列名)
分组和排序的语法相同,只是分组不接受方向作为数组的最后一个参数(不存在 ASC
, DESC
, NULLS FIRST
等).
你还可以将字符串直接传递给 group
,该字符串将直接(普通)包含在生成的 SQL 中. 请谨慎使用,请勿与用户生成的内容一起使用.
Project.findAll({ group: 'name' });
限制和分页#
使用 limit
和 offset
参数可以进行 限制/分页:
Project.findAll({ limit: 10 });
Project.findAll({ offset: 8 });
Project.findAll({ offset: 5, limit: 5 });
通常这些与 order
参数一起使用.
实用方法#
Sequelize 还提供了一些实用方法.
count
#
count
方法仅计算数据库中元素出现的次数.
console.log(`这有 ${await Project.count()} 个项目`);
const amount = await Project.count({
where: {
id: {
[Op.gt]: 25
}
}
});
console.log(`这有 ${amount} 个项目 id 大于 25`);
max
, min
和 sum
#
Sequelize 还提供了 max,min 和 sum 便捷方法.
假设我们有三个用户,分别是10、5和40岁.
await User.max('age');
await User.max('age', { where: { age: { [Op.lt]: 20 } } });
await User.min('age');
await User.min('age', { where: { age: { [Op.gt]: 5 } } });
await User.sum('age');
await User.sum('age', { where: { age: { [Op.gt]: 5 } } });