如何在Node.js上使用Sequelize进行连接查询

Jos*_*osa 81 mysql orm node.js sequelize.js

我正在使用sequelize ORM; 一切都很棒,很干净,但是当我将它用于join查询时,我遇到了问题.我有两个模型:用户和帖子.

var User = db.seq.define('User',{
    username: { type: db.Sequelize.STRING},
    email: { type: db.Sequelize.STRING},
    password: { type: db.Sequelize.STRING},
    sex : { type: db.Sequelize.INTEGER},
    day_birth: { type: db.Sequelize.INTEGER},
    month_birth: { type: db.Sequelize.INTEGER},
    year_birth: { type: db.Sequelize.INTEGER}

});

User.sync().success(function(){
    console.log("table created")
}).error(function(error){
    console.log(err);
})


var Post = db.seq.define("Post",{
    body: { type: db.Sequelize.TEXT },
    user_id: { type: db.Sequelize.INTEGER},
    likes: { type: db.Sequelize.INTEGER, defaultValue: 0 },

});

Post.sync().success(function(){
    console.log("table created")
}).error(function(error){
    console.log(err);
})
Run Code Online (Sandbox Code Playgroud)

我想要一个回复帖子的查询,其中包含用户的信息.在原始查询中,我得到了这个:

db.seq.query('SELECT * FROM posts, users WHERE posts.user_id = users.id ').success(function(rows){
            res.json(rows);
        });
Run Code Online (Sandbox Code Playgroud)

我的问题是如何更改代码以使用ORM样式而不是SQL查询?

Jan*_*ier 103

User.hasMany(Post, {foreignKey: 'user_id'})
Post.belongsTo(User, {foreignKey: 'user_id'})

Post.find({ where: { ...}, include: [User]})
Run Code Online (Sandbox Code Playgroud)

哪个会给你

SELECT
  `posts`.*,
  `users`.`username` AS `users.username`, `users`.`email` AS `users.email`,
  `users`.`password` AS `users.password`, `users`.`sex` AS `users.sex`,
  `users`.`day_birth` AS `users.day_birth`,
  `users`.`month_birth` AS `users.month_birth`,
  `users`.`year_birth` AS `users.year_birth`, `users`.`id` AS `users.id`,
  `users`.`createdAt` AS `users.createdAt`,
  `users`.`updatedAt` AS `users.updatedAt`
FROM `posts`
  LEFT OUTER JOIN `users` AS `users` ON `users`.`id` = `posts`.`user_id`;
Run Code Online (Sandbox Code Playgroud)

与您发布的内容相比,上面的查询可能看起来有点复杂,但它的作用基本上只是对users表的所有列进行别名,以确保在返回时将它们放入正确的模型中,而不是与posts模型混淆

除此之外,您会注意到它执行了JOIN而不是从两个表中进行选择,但结果应该相同

进一步阅读:

  • 如果我只想加入1984年出生的用户怎么办?在SQL中我会这样做:`SELECT*FROM posts JOIN users ON users.id = posts.user_id WHERE users.year_birth = 1984` (6认同)
  • @antew 当您调用`User.hasMany(Post)` 时,您将方法/属性添加到User 对象,而当您调用`Post.belongsTo(User)` 时,您将方法添加到Post 类。如果您总是从一个方向调用(例如 User.getPosts()),那么您不需要向 Post 对象添加任何内容。但是两边都有方法是很好的。 (2认同)

Rya*_*ton 101

虽然接受的答案在技术上不是错误的,但它没有回答原始问题,也没有回答评论中的后续问题,而这正是我来到这里寻找的.但我明白了,所以这里.

如果要查找具有用户(并且只有具有用户的用户)的所有帖子,其中SQL将如下所示:

SELECT * FROM posts INNER JOIN users ON posts.user_id = users.id
Run Code Online (Sandbox Code Playgroud)

这在语义上与OP的原始SQL相同:

SELECT * FROM posts, users WHERE posts.user_id = users.id
Run Code Online (Sandbox Code Playgroud)

那就是你想要的:

Posts.findAll({
  include: [{
    model: User,
    required: true
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

设置为true是生成内部联接的关键.如果你想要一个左外连接(你获得所有帖子,无论是否有用户链接),然后将所需更改为false,或将其保持关闭,因为这是默认值:

Posts.findAll({
  include: [{
    model: User,
//  required: false
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

如果要查找属于出生年份为1984年的用户的所有帖子,您需要:

Posts.findAll({
  include: [{
    model: User,
    where: {year_birth: 1984}
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

请注意,只要在其中添加where子句,默认情况下为true.

如果你想要所有帖子,无论是否有用户附加,但是如果有用户,那么只有1984年出生的用户,然后将所需的字段添加回来:

Posts.findAll({
  include: [{
    model: User,
    where: {year_birth: 1984}
    required: false,
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

如果你想要名称为"Sunshine"的所有帖子,并且只有它属于1984年出生的用户,你可以这样做:

Posts.findAll({
  where: {name: "Sunshine"},
  include: [{
    model: User,
    where: {year_birth: 1984}
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

如果您想要名称为"Sunshine"的所有帖子,并且只有当它属于与帖子上的post_year属性匹配的同一年出生的用户时,您才会这样做:

Posts.findAll({
  where: {name: "Sunshine"},
  include: [{
    model: User,
    where: ["year_birth = post_year"]
   }]
}).then(posts => {
  /* ... */
});
Run Code Online (Sandbox Code Playgroud)

我知道,有人会在他们出生的那一年发表一篇文章是没有道理的,但这仅仅是一个例子 - 坚持下去.:)

我(主要)从这个doc中找到了这个:

  • 非常感谢,这很有用 (6认同)
  • 是的,很好的答案 (5认同)
  • 这个答案是如此详尽,应该在Sequelize的文档中联系起来 (3认同)
  • 这是一个惊人的答案,谢谢你这么匹配 (3认同)
  • 那么在你的续集中什么代表`posts.user_id = users.id`?谢谢 (2认同)
  • @hanzichi “posts.user_id”列与“users.id”列匹配是在您的续集模型定义中设置的。 (2认同)
  • 我一直在寻找的很棒的教程。非常感谢 (2认同)
  • @JasonG你想要“属性”键。在此文档中搜索“属性”:https://sequelize.org/master/manual/model-querying-basics.html (2认同)

小智 7

Model1.belongsTo(Model2, { as: 'alias' })

Model1.findAll({include: [{model: Model2  , as: 'alias'  }]},{raw: true}).success(onSuccess).error(onError);
Run Code Online (Sandbox Code Playgroud)