mysql一对一消息优化表查询

mysql一对一消息优化表查询,mysql,sql,database,database-design,database-schema,Mysql,Sql,Database,Database Design,Database Schema,我使用了以下模式: -- store messages between users CREATE TABLE chat_messages ( id BIGINT UNSIGNED NOT NULL AUTO_INCREMENT, sender_id BIGINT UNSIGNED NOT NULL, recipient_id BIGINT UNSIGNED NOT NULL, message_body TEXT, read_at DATETIME DEFA

我使用了以下模式:

-- store messages between users
CREATE TABLE chat_messages (
    id BIGINT UNSIGNED NOT NULL AUTO_INCREMENT,
    sender_id BIGINT UNSIGNED NOT NULL,
    recipient_id BIGINT UNSIGNED NOT NULL,
    message_body TEXT,
    read_at DATETIME DEFAULT NULL,
    created_at DATETIME NOT NULL,
    PRIMARY KEY(id),
    CONSTRAINT FK_messages_1 FOREIGN KEY (sender_id) REFERENCES profiles (id),
    CONSTRAINT FK_messages_2 FOREIGN KEY (recipient_id) REFERENCES profiles (id)
) ENGINE = INNODB;
我需要对其运行查询以获取用户的唯一活动聊天列表,我有以下查询:

SELECT * FROM chat_messages WHERE recipient_id = :user_id GROUP BY recipient_id ORDER BY created_at DESC
它返回分组的行,但是:

  • 返回的行包括消息体,但它不是最新的消息(按照需要按先前排序的数据集对其进行分组的思路)

  • 此外,如果用户向另一个用户发送了消息,但尚未收到回复,则不会将其包括在列表中(在这种情况下,还应考虑发件人的id)


  • 有没有建议使用以性能为中心的查询来实现这一切?谢谢:)

    您可以使用一个不存在的
    查询来做您想做的事情:

    SELECT cm.*
    FROM chat_messages cm
    WHERE recipient_id = :user_id AND
          NOT EXISTS (SELECT 1
                      FROM chat_messages cm2
                      WHERE cm2.recipient_id = cm.recipient_id AND
                            cm2.created_at > cm.created_at
                     )
    ORDER BY created_at DESC;
    

    为了提高性能,您应该在
    聊天信息(收件人id,创建地址)
    上有一个索引这里有一个查询,它应该满足您的需要:

    SELECT a.sender_id, a.recipient_id, a.message_body
    FROM chat_messages a
    INNER JOIN (
      SELECT IF(recipient_id=1, sender_id, recipient_id) user_id,
        MAX(created_at) created_at
      FROM chat_messages
      WHERE 1 IN (recipient_id, sender_id)
      GROUP BY IF(recipient_id=1, sender_id, recipient_id)
    ) recent
    ON 
      (
        (a.sender_id = recent.user_id AND a.recipient_id = 1)
      OR
        (a.recipient_id = recent.user_id AND a.sender_id = 1)
      )
      AND a.created_at = recent.created_at
    
    内部联接计算对话伙伴用户ID和最近消息的日期。它与整个表连接,以搜索相应的消息并检索其内容

    如果您在
    (发送者id、接收者id、创建地点)
    上创建索引,它可以用于支持执行。然而,这个问题并不是那么简单


    我也准备了一份电子邮件。

    事实上,你也可以随机收到最新的邮件。或介于两者之间的任何消息体。这是一个不幸的结果(引用:“服务器可以从每个组中自由选择任何值,因此,除非它们相同,否则选择的值是不确定的。”)。其他DBMS做了正确的事情,禁止选择列表中既没有聚合也没有在GROUP BY中提及的列。Gordon,感谢您的回答-非常感谢-它适用于第1点。(它获取最后一条消息),任何可能的扩展都要执行第2点。作为此查询的一部分?还是最好引入一个“chat_线程”表并在chat_消息中引用为FK?