SQL:动态扩展搜索条件

Posted

技术标签:

【中文标题】SQL:动态扩展搜索条件【英文标题】:SQL: dynamically expand search criteria 【发布时间】:2014-07-03 14:47:56 【问题描述】:

我有一个表格,其中包含带有坐标(纬度、经度)的本地企业 我需要获取给定半径内的所有企业,为此我使用了一个实时计算距离的查询

select b.* from `businesses` as b
where
(
1.609344 * 3956 * 2 * ASIN(
    SQRT( 
            POWER(SIN((lat - b.lat) *  pi()/180 / 2), 2) +
            COS(lat * pi()/180) * COS(b.lat * pi()/180) * 
            POWER(SIN((lng -b.lng) * pi()/180 /-2), 2) 
        )
    )
) <= radius

现在我需要扩大半径,以防该地区的企业太少, 所以假设在给定的半径内只有 10 家企业,我需要动态扩大半径,直到我得到 50

【问题讨论】:

你能去掉 WHERE 子句中的 但如果超过 50 个,我需要全部查看。 哦,我明白了……在 SQL Server 中,我可能会使用递归 CTE 来处理类似的事情,但我认为 mysql 中没有类似的构造。在您的情况下,我可能会做的是创建一个带有临时表的存储过程,并用第一个结果填充它。如果它小于 50,那么我会在查询中循环增加半径,直到有 50,然后返回那个临时表。 【参考方案1】:

只需计算到给定点的距离并使用它来排序,然后将结果数限制为 50。

如果该半径内立即有超过 50 个结果,如果您想要超过 50 个结果,这将变得有点棘手。

在这种情况下,您可以执行类似 'if distance

这看起来像这样:

WITH t1 AS
(
    SELECT b.*,
    1.609344 * 3956 * 2 * ASIN(
                SQRT( 
                    POWER(SIN((lat - b.lat) *  pi()/180 / 2), 2) +
                    COS(lat * pi()/180) * COS(b.lat * pi()/180) * POWER(SIN((lng -b.lng) * pi()/180 / 2), 2) 
                )
            )
    ) AS distance
    FROM businesses as b
), t2 AS
(
     SELECT *, @rownum := @rownum + 1 AS row_num
     FROM t1, (SELECT @rownum := 0) r
     ORDER BY distance
)
SELECT * FROM t2
WHERE distance < ? OR row_num < ?

【讨论】:

我没有提到我正在使用MySql,它不支持ROW_NUMBER()

以上是关于SQL:动态扩展搜索条件的主要内容,如果未能解决你的问题,请参考以下文章

笔记:MyBatis 动态SQL

扩展报表-JavaSet

Oracle 在执行动态生成的 SQL 查询时进行扩展

[转]使用exec和sp_executesql动态执行SQL语句

在T-SQL中具有可变数量的搜索条件的LIKE运算符

在sql的where条件下怎么加判断语句?