陌上桑花开花 2019-06-26 11:23:31 12094 收藏 8
分类专栏: database
版权
环境:idea+mysql5.7
mysql any_value()函数用法:
地域表结构:省份和城市,城市等级
业务要求:查询所有省份:
方法一:distinct排除重复
SELECT
DISTINCT(province_code),
province_name
FROM
t_mip_base_area
方法二:group by 根据身份编码分组
SELECT
province_code,
any_value(province_name)
FROM t_mip_base_area
GROUP BY province_code
若这样写:
SELECT
province_code,
province_name
FROM t_mip_base_area
GROUP BY province_code
则报:
总结:
1.MySQL5.7之后,sql_mode中ONLY_FULL_GROUP_BY模式默认设置为打开状态。
2.ONLY_FULL_GROUP_BY的语义就是确定select target list中的所有列的值都是明确语义,简单的说来,在此模式下,target list中的值要么是来自于聚合函数(sum、avg、max等)的结果,要么是来自于group by list中的表达式的值
3.MySQL提供了any_value()函数来抑制ONLY_FULL_GROUP_BY值被拒绝
4.any_value()会选择被分到同一组的数据里第一条数据的指定列值作为返回数据
您可以使用MIN或MAX聚合函数代替ANY_VALUE。
或者,您可以考虑不设置ONLY_FULL_GROUP_BY SQL模式,默认情况下为MySql 5.7设置,并且负责您与MySql 5.6的不同之处。然后,您可以延迟更新查询,直到将所有环境迁移到MySql 5.7。
这两个中哪一个是更好的选择,值得商榷,但从长远来看,最好调整您的查询,使其符合ONLY_FULL_GROUP_BY规则。使用MIN或MAX肯定可以用来做到这一点。
https://www.thinbug.com/q/37089347
时间:2016-05-07 13:47:25
标签: mysql group-by aggregate-functions mysql-error-1055
目前正在开发中使用mysql 5.7,在生产中使用5.6。每次我在开发中使用组运行查询时都会出现一些错误,例如“错误代码:1055. SELECT列表的表达式#1不在GROUP BY中”
这是查询。
<script language="JavaScript"> var sPath=window.location.pathname; var sPage = sPath.substring(sPath.lastIndexOf('/') + 1); var sHost = 'host.html' ; if(spage = sHost){ alert('orange'); } </script>为了解决这个问题,我使用5.7 ANY_VALUE中的mysql函数,但主要问题是它在mysql 5.6中不可用
因此,如果我修复sql语句进行开发,我将在生产中出错。
你知道mysql 5.6中ANY_VALUE函数的任何解决方案或polifill吗?
3 个答案:
答案 0 :(得分:16)
您可以使用MIN或MAX聚合函数代替ANY_VALUE。
或者,您可以考虑不设置ONLY_FULL_GROUP_BY SQL模式,默认情况下为MySql 5.7设置,并且负责您与MySql 5.6的不同之处。然后,您可以延迟更新查询,直到将所有环境迁移到MySql 5.7。
这两个中哪一个是更好的选择,值得商榷,但从长远来看,最好调整您的查询,使其符合ONLY_FULL_GROUP_BY规则。使用MIN或MAX肯定可以用来做到这一点。
答案 1 :(得分:13)
你误导了notorious nonstandard MySQL extension to GROUP BY。标准SQL将始终拒绝您的查询,因为您提到的不是聚合的列,并且在GROUP BY中未提及。在您的开发系统中,您正尝试使用ANY_VALUE()来解决这个问题。
在制作中,您可以关闭ONLY_FULL_GROUP_BY MySQL Mode.尝试执行this:
SET @mode := @@SESSION.sql_mode; SET SESSION sql_mode = ''; /* your query here */ SET SESSION sql_mode = @mode;这将允许MySQL接受您的查询。
但是看,你的查询并不正确。当你可以说服它运行时,它会从images表中返回一个随机选择的行。这种不确定性常常会给用户和您的技术支持人员造成混淆。
为什么不让查询更好,所以它选择一个特定的图像。如果您的images表格中包含自动增量id列,则可以选择“第一张”图片。
SELECT c.id, c.name, i.* FROM countries c LEFT JOIN ( SELECT MIN(id) id, country_id FROM images GROUP BY country_id ) first ON c.id = first.country_id LEFT JOIN images i ON first.id = i.id每个国家/地区将返回一行,并显示可预测的图像。
答案 2 :(得分:4)
几十年来,您可以编写在标准SQL中无效但完全有效的mysql
的查询
在标准SQL中,包含GROUP BY子句的查询无法引用 选择列表中未分配的非聚合列 GROUP BY子句。例如,此查询在标准SQL中是非法的 因为选择列表中的非聚合名称列不会 出现在GROUP BY:
SELECT o.custid,c.name,MAX(o.payment)FROM订购AS o,客户 as c WHERE o.custid = c.custid GROUP BY o.custid;对于查询 合法,名称列必须从选择列表中删除或 在GROUP BY子句中命名。
MySQL扩展了GROUP BY的标准SQL使用,以便选择列表 可以引用GROUP BY子句中未命名的非聚合列。
这来自GROUP BY上的Mysql 5.6手册页面。如果您查看5.7.6的同一页面,您会发现事情已经发生了变化。并且发生了巨大变化!
该页面也为您提供了解决方案。禁用ONLY_FULL_GROUP_BY这将使您的旧5.6查询可以在5.7.6上运行(从查询中删除ANY_VALUE,因为它在5.7.6中不可用,而是使用ONLY_FULL_GROUP_BY)。
相关问题
在SQL中,是否有类似于:WHERE x = ANY_VALUE?是否有emacs功能可以访问上次编辑?MYSQL:列IN(Any_value)使用PHP 5.6和PDO是否需要转义字符?mysql 5.6有ANY_VALUE功能吗?Msgpack中是否有版本控制功能Sequelize.fn(`ANY_VALUE`,sequelize.col(`col`));不返回任何数据在mysql 5.6中,ANY_VALUE()是否有其他选择?是否有5.6的Laravel集体文档MariaDb不支持ANY_VALUE()函数最新问题
为什么我的登录按钮没有输入密码?我不明白重构此方法以将其认知复杂度从19降低到15为什么一直跳过名称输入?Java:将类对象转换为地图的最快方法是什么在设置nth-child(odd)并设置'.sayari:nth-child(odd){clear:both;}'之后,为什么div元素不会浮动到右上角?Javascript,新创建的段不起作用Unity中的影子如何使该导航栏居中?我可以获取将代码发送到LCD的简易代码吗?在reactjs和nextjs构造函数中获取参考错误:
小李小李彬彬有礼 2019-05-28 16:21:10 23311 收藏 19
分类专栏: 数据库 文章标签: any_value()
版权
MySQL5.7之后,sql_mode中ONLY_FULL_GROUP_BY模式默认设置为打开状态。
ONLY_FULL_GROUP_BY的语义就是确定select target list中的所有列的值都是明确语义,简单的说来,在此模式下,target list中的值要么是来自于聚合函数(sum、avg、max等)的结果,要么是来自于group by list中的表达式的值。
网上有很多通过修改sql_mode的方式来解决此问题。
但除此方法,MySQL也提供了any_value()函数来抑制ONLY_FULL_GROUP_BY值被拒绝
举例:表里存了员工姓名,部门名称,员工薪资信息
由于在通过部门分组的时候,ename的值出现的碰撞,以第一部门举例,通过group by后,第一部门会以一条数据的形式展示,但是部门里有两个员工姓名是不一样的,那么应该展示哪个呢?
额外补充:如果是5.7版本以上的话,即使所有记录的ename的值都一样,也会出现这个错误。因为上面提到了如果ONLY_FULL_GROUP_BY模式开启,那么select target list中的所有列的值都必须有明确语义。其实例子中的ename对我们来说意义不大,我们完全可以将其从target list中移除。但是如果真实的业务场景中必须要查询这一项,那我们就可以使用any_value()函数。
尝试用一下any_value()函数
根据结果猜想:any_value()会选择被分到同一组的数据里第一条数据的指定列值作为返回数据。
这次查询第一部门的ename展示了一号员工,那我们换一下部门里员工的顺序,重新查询一下,验证我们的猜想
根据结果可知,我们的猜想是正确的。
https://www.itranslater.com/qa/details/2109775246877262848
现在,你可能会问,为什么不推荐使用ANY_VALUE?因为MySQL并不确切知道要检索的分组记录的值,并且通过使用此函数,您要求它获取它们中的任何一个(在这种情况下,获取名称= John的第一条记录的电子邮件)。确切地说,我无法想出为什么你希望这种行为存在的想法。 如果你不理解我,请阅读更多关于如何在MySQL中进行分组的工作,这很简单。
到最后,这是一个更简单但有效的查询。 如果要根据可用年龄查询总用户数,可能需要记下此查询
SELECT `age`, COUNT(`age`) FROM `users` GROUP BY `age`;根据MySQL规则,这完全有效。 等等。 重要的是要了解问题到底是什么,然后记下解决方案。