简体   繁体   English

BigQuery - 查找最近的区域

[英]BigQuery - Find the closest region

I have two tables, and for each region in A, I want to find the closest regions in B.我有两个表,对于 A 中的每个区域,我想找到 B 中最近的区域。

A:
------------------------
ID | Start | End | Color 
------------------------
 1 |  400  | 500 | White
------------------------
 1 |  10   | 20  | Red 
------------------------
 2 |   2   |  10 | Blue 
------------------------
 4 |   88  |  90 | Color 
------------------------

B:
------------------------
ID | Start | End | Name 
------------------------
 1 |  1    | 2   | XYZ1 
------------------------
 1 |  50   | 60  | XYZ4 
------------------------
 2 |  150  | 160 | ABC1 
------------------------
 2 |  50   | 60  | ABC2 
------------------------
 4 |  100  | 120 | EFG 
------------------------

RS:
---------------------------------------
ID | Start | End | Color | Closest Name
---------------------------------------
 1 |  400  | 500 | White |   XYZ4
---------------------------------------
 1 |  10   | 20  | Red   |   XYZ1
---------------------------------------
 2 |   2   |  10 | Blue  |   ABC2
---------------------------------------
 4 |   88  |  90 | Color |   EFG
---------------------------------------

Currently, I first find min distance by joining two tables:目前,我首先通过连接两个表来找到最小距离:

MinDist Table: MinDist表:

SELECT   A.ID,   A.Start,  A.End,   
MIN(CASE  
WHEN (ABS(A.End-B.Start)>=ABS(A.Start - B.End)) 
THEN ABS(A.Start-B.End)     
ELSE ABS(A.End - B.Start) 
END) AS distance 
FROM ( Select A ... ) 
Join B On A.ID=B.ID) 
Group By A.ID,   A.Start,  A.End

Then recompute distance for by joining table A and B again, GlobDist Table (Note, the query retrieves B.Name in this case):然后通过再次连接表 A 和 B 重新计算距离, GlobDist表(注意,在这种情况下,查询检索 B.Name):

SELECT   A.ID,   A.Start,  A.End,   
CASE  
WHEN (ABS(A.End-B.Start)>=ABS(A.Start - B.End)) 
THEN ABS(A.Start-B.End)     
ELSE ABS(A.End - B.Start) 
END AS distance,
B.Name 
FROM ( Select A ... ) 
Join B On A.ID=B.ID) 

Finally join these two tables MinDist and GlobDist Tables on最后加入这两个表 MinDist 和 GlobDist 表

GlobDist.ID= MinDist.ID, 
GlobDist.Start=MinDist.Start, 
GlobDist.End= MinDist.End, 
GlobDist.distance= MinDist.distance.

I tested ROW_NUMBER() and PARTITION BY over (ID, Start, End), but it took much longer.我测试了 ROW_NUMBER() 和 PARTITION BY over (ID, Start, End),但花了更长的时间。 So, what's the fastest and most efficient way of solving this problem?那么,解决这个问题的最快和最有效的方法是什么? How can I reduce duplicate computation?如何减少重复计算?

Thanks!谢谢!

Below solution is for BigQuery Standard SQL and as simple and short as below以下解决方案适用于 BigQuery Standard SQL,如下所示

#standardSQL
SELECT a_id, a_start, a_end, color,  
  ARRAY_AGG(name ORDER BY POW(ABS(a_start - b_start), 2) + POW(ABS(a_end - b_end), 2) LIMIT 1)[SAFE_OFFSET(0)] name
FROM A JOIN B ON a_id = b_id
GROUP BY a_id, a_start, a_end, color
-- ORDER BY a_id

You can test / play with above using dummy data in your question您可以在问题中使用虚拟数据测试/玩上面

#standardSQL
WITH A AS (
  SELECT 1 a_id, 400 a_start, 500 a_end, 'White' color UNION ALL
  SELECT 1, 10,  20  , 'Red' UNION ALL
  SELECT 2, 2,   10, 'Blue' UNION ALL
  SELECT 4, 88,  90, 'Color'
), B AS (
  SELECT 1 b_id, 1 b_start, 2 b_end, 'XYZ1' name UNION ALL
  SELECT 1, 50, 60,  'XYZ4' UNION ALL
  SELECT 2, 150, 160,'ABC1' UNION ALL
  SELECT 2, 50, 60,  'ABC2' UNION ALL
  SELECT 4, 100, 120,'EFG'
)
SELECT a_id, a_start, a_end, color,  
  ARRAY_AGG(name ORDER BY POW(ABS(a_start - b_start), 2) + POW(ABS(a_end - b_end), 2) LIMIT 1)[SAFE_OFFSET(0)] name
FROM A JOIN B ON a_id = b_id
GROUP BY a_id, a_start, a_end, color
ORDER BY a_id  

with result as below结果如下

Row a_id    a_start a_end   color   name     
1   1       400     500     White   XYZ4     
2   1       10      20      Red     XYZ1     
3   2       2       10      Blue    ABC2     
4   4       88      90      Color   EFG

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM