jjzjj

php - 通过给定字段通过 UNION 组加入 2 个表,并仅获取该组的最新信息

coder 2023-10-24 原文

我正在一个小仪表板中构建一个电子邮件收件箱。是使用其他 API 的自定义电子邮件客户端。我准备DB结构的方式如下:

传入

╔════╦════════════════════╦═══════════════════╦═════════════════════════════════════════╦══════════════════════╗
║ id ║     email_from     ║     email_to      ║               email_body                ║         date         ║
╠════╬════════════════════╬═══════════════════╬═════════════════════════════════════════╬══════════════════════╣
║  1 ║ stack@overflow.com ║ mitch@example.com ║ Your question has a downvote for newbie ║ 2017-12-28T00:00:00Z ║
║  2 ║ stack@exchange.com ║ mitch@example.com ║ You gotta learn SQL, buddy :/           ║ 2017-12-28T00:01:43Z ║
║  3 ║ stack@overflow.com ║ mitch@example.com ║ Guess what? 42                          ║ 2017-12-28T00:05:00Z ║
║  3 ║ stack@overflow.com ║ mitch@example.com ║ This is a bot, stop responding          ║ 2017-12-28T00:10:00Z ║
╚════╩════════════════════╩═══════════════════╩═════════════════════════════════════════╩══════════════════════╝

传出

╔════╦═══════════════════╦════════════════════╦════════════════════════════════════╦══════════════════════╗
║ id ║    email_from     ║      email_to      ║             email_body             ║         date         ║
╠════╬═══════════════════╬════════════════════╬════════════════════════════════════╬══════════════════════╣
║  1 ║ mitch@example.com ║ stack@overflow.com ║ That is bad news                   ║ 2017-12-28T00:00:50Z ║
║  2 ║ mitch@example.com ║ stack@exchange.com ║ I know :(                          ║ 2017-12-28T00:01:45Z ║
║  3 ║ mitch@example.com ║ stack@overflow.com ║ Answer to the Ultimate Question... ║ 2017-12-28T00:07:42Z ║
╚════╩═══════════════════╩════════════════════╩════════════════════════════════════╩══════════════════════╝

所以,我想做一个像 Gmail 这样的小收件箱,您可以在其中看到所有电子邮件的列表,按电子邮件地址(或对话)分组并按日期排序,在顶部显示最新的,在每个对话中,显示仅该对话的最后一条消息。


我的查询尝试

SELECT `email_from` as `thread_email`, `email_body`, `date`
FROM (
SELECT `email_from`, `email_body`, `date` FROM `incoming`
UNION
SELECT `email_to` as `email_from`, `email_body`, `date` FROM `outgoing`
) AS t_union

GROUP BY `email_from`
ORDER BY `date` DESC

结果:

╔════════════════════╦═════════════════════════════════════════╦══════════════════════╗
║    thread_email    ║               email_body                ║         date         ║
╠════════════════════╬═════════════════════════════════════════╬══════════════════════╣
║ stack@exchange.com ║ You gotta learn SQL, buddy :/           ║ 2017-12-28T00:01:43Z ║
║ stack@overflow.com ║ Your question has a downvote for newbie ║ 2017-12-28T00:00:00Z ║
╚════════════════════╩═════════════════════════════════════════╩══════════════════════╝

它仅显示来自 incoming 表的结果。


我的预期结果

  • 按电子邮件地址分组,这代表一个对话(称为 thread_email,基本上每个 thread_email 一行)
  • 每一行(或 session ,或 thread_email)仅显示该 thread_email 的最新消息(最后一条消息可能是传入或传出)
  • 按日期排序(最近的在前)
╔════════════════════╦════════════════════════════════╦══════════════════════╗
║    thread_email    ║           email_body           ║         date         ║
╠════════════════════╬════════════════════════════════╬══════════════════════╣
║ stack@overflow.com ║ This is a bot, stop responding ║ 2017-12-28T00:10:00Z ║
║ stack@exchange.com ║ I know :(                      ║ 2017-12-28T00:01:45Z ║
╚════════════════════╩════════════════════════════════╩══════════════════════╝

SQL fiddle :http://sqlfiddle.com/#!9/ed9642/4

这是实现这一目标的新手方法吗?对另一种数据库结构有什么建议,这种结构更简单但也更清晰有序?

最佳答案

正如 Gordon 所说,如果 group by 你的数据,将数据减少到一行,但如果你只使用 order by 子句,你仍然会得到重复的数据, 它将从联合表中获取所有数据。

group by 子句也有一个 having 子句,但它似乎只适用于数字,比如 count(*) > 2 和其他东西。

PostgreSQL我们有一个叫做 array_agg 的东西,它将一些数据转换为数组,然后我们可以将其作为数组访问。

在 MySQL 中,我发现的唯一类似的是 group_concat将所有数据与相同的分组语句转换为字符串中用','连接的语句,所以,idk如果这解决了你的问题,但你可以尝试这个并在应用程序端处理结果,而不是在SQL端处理。

你可以这样使用:

select email_from, group_concat(email_body), date from (
  select email_from, email_body, date from incoming union all
  select email_to as email_from, email_body, date from outgoing
) as t_union group by email_from order by date desc;

或者,在最后一种情况下,以非优化方式执行:

select email_from, email_body, date from (
  select email_from, email_body, date from incoming union all
  select email_to as email_from, email_body, date from outgoing
) as t_union_1 where date in (
  select max(date) from (
    select email_from, email_body, date from incoming union all
    select email_to as email_from, email_body, date from outgoing
  ) as t_union group by email_from order by date desc
);

关于php - 通过给定字段通过 UNION 组加入 2 个表,并仅获取该组的最新信息,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48000660/

有关php - 通过给定字段通过 UNION 组加入 2 个表,并仅获取该组的最新信息的更多相关文章

  1. ruby-on-rails - Rails 常用字符串(用于通知和错误信息等) - 2

    大约一年前,我决定确保每个包含非唯一文本的Flash通知都将从模块中的方法中获取文本。我这样做的最初原因是为了避免一遍又一遍地输入相同的字符串。如果我想更改措辞,我可以在一个地方轻松完成,而且一遍又一遍地重复同一件事而出现拼写错误的可能性也会降低。我最终得到的是这样的:moduleMessagesdefformat_error_messages(errors)errors.map{|attribute,message|"Error:#{attribute.to_s.titleize}#{message}."}enddeferror_message_could_not_find(obje

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  4. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  5. ruby - 如何离开加入Arel? - 2

    Arel3.0.2提供了两个类来指定连接类型:Arel::Nodes::InnerJoin和Arel::Nodes::OuterJoin并使用InnerJoin默认。foo=Arel::Table.new('foo')bar=Arel::Table.new('bar')foo.join(bar,Arel::Nodes::InnerJoin)#innerfoo.join(bar,Arel::Nodes::OuterJoin)#outerfoo.join(bar,???)#left如果要生成左连接,如何连接两个表? 最佳答案 你可以使用

  6. ruby-on-rails - 如何验证非模型(甚至非对象)字段 - 2

    我有一个表单,其中有很多字段取自数组(而不是模型或对象)。我如何验证这些字段的存在?solve_problem_pathdo|f|%>... 最佳答案 创建一个简单的类来包装请求参数并使用ActiveModel::Validations。#definedsomewhere,atthesimplest:require'ostruct'classSolvetrue#youcouldevencheckthesolutionwithavalidatorvalidatedoerrors.add(:base,"WRONG!!!")unlesss

  7. ruby-on-rails - form_for 中不在模型中的自定义字段 - 2

    我想向我的Controller传递一个参数,它是一个简单的复选框,但我不知道如何在模型的form_for中引入它,这是我的观点:{:id=>'go_finance'}do|f|%>Transferirde:para:Entrada:"input",:placeholder=>"Quantofoiganho?"%>Saída:"output",:placeholder=>"Quantofoigasto?"%>Nota:我想做一个额外的复选框,但我该怎么做,模型中没有一个对象,而是一个要检查的对象,以便在Controller中创建一个ifelse,如果没有检查,请帮助我,非常感谢,谢谢

  8. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

  9. ruby - 通过 RVM (OSX Mountain Lion) 安装 Ruby 2.0.0-p247 时遇到问题 - 2

    我的最终目标是安装当前版本的RubyonRails。我在OSXMountainLion上运行。到目前为止,这是我的过程:已安装的RVM$\curl-Lhttps://get.rvm.io|bash-sstable检查已知(我假设已批准)安装$rvmlistknown我看到当前的稳定版本可用[ruby-]2.0.0[-p247]输入命令安装$rvminstall2.0.0-p247注意:我也试过这些安装命令$rvminstallruby-2.0.0-p247$rvminstallruby=2.0.0-p247我很快就无处可去了。结果:$rvminstall2.0.0-p247Search

  10. ruby-on-rails - Enumerator.new 如何处理已通过的 block ? - 2

    我在理解Enumerator.new方法的工作原理时遇到了一些困难。假设文档中的示例:fib=Enumerator.newdo|y|a=b=1loopdoy[1,1,2,3,5,8,13,21,34,55]循环中断条件在哪里,它如何知道循环应该迭代多少次(因为它没有任何明确的中断条件并且看起来像无限循环)? 最佳答案 Enumerator使用Fibers在内部。您的示例等效于:require'fiber'fiber=Fiber.newdoa=b=1loopdoFiber.yieldaa,b=b,a+bendend10.times.m

随机推荐