如何在where子句中使用带有可选包含的运算符?

时间:2017-11-13 11:45:48

标签: mysql sequelize.js

我有以下型号:

AuthorModel.hasMany(BookModel);
BookModel.belongsTo(AuthorModel);

有些作者没有书。

我想选择一个作者的名字或书名与搜索字符串匹配。

我可以通过以下声明来实现这一点,但仅适用于BookModel中有书籍的作者

       Author.findOne({
         include: [{
            model: Book,
            where: {
              [Op.or]: [
                  {'$author.name$': 'search string'},
                  { title: 'search string'}
                 ]
               },
             }]
           })

这或多或少地为我提供了以下mysql查询:

SELECT 
    `author`.`name`,
    `book`.`title`
FROM `author` INNER JOIN `book` 
     ON `author`.`id` = `book`.`authorId`
     AND ( `author`.`name` = 'search string' OR `book`.`title` = 'search string');

这里的问题是,如果作者没有书籍,那么结果是空的。即使有作者符合搜索条件。

我尝试将包含设置为required: false,其中包含left outer join。在这种情况下,我得到一些不匹配的结果。省略了where子句。

如何更改sequelize查询或正确的mysql查询?

1 个答案:

答案 0 :(得分:3)

MySql查询应该类似于

SELECT 
    `author`.`name`,
    `book`.`title`
FROM `author` LEFT JOIN `book` 
     ON `author`.`id` = `book`.`authorId`
WHERE ( `author`.`name` = 'search string' OR `book`.`title` = 'search string')

请注意这里的过滤条件是WHERE而不是JOIN ... ON子句

的一部分

根据Top level where with eagerly loaded models上的示例,你的squizzle查询可能应该像

Author.findOne({
    where: {
          [Op.or]: [
              {'$author.name$': 'search string'},
              { '$Book.title$': 'search string'}
             ]
           },
    include: [{
        model: Book,           
        required: false
       }]})