SQL使用长/纬度计算距离&转换Int。到十进制

时间:2013-02-01 21:09:19

标签: sql floating-point sql-server-2008-r2 latitude-longitude

原帖:Changing integer to floating point and adding decimal point

所以这是后面的故事:我正在使用一个接收车辆GPS坐标的数据库,我需要根据经度和纬度值计算出这些车辆的里程数。但是,在数据库中,经度/纬度列都是“Int / Not null”并且没有小数位。所以他们看起来像这样:

Latitude Longitude 
36158500 115949833
36340000 115914667
36153488 115944875

我需要它看起来像这样:

Latitude Longitude 
36.158500 115.949833  
36.340000 115.914667 
36.153488 115.944875

在Tim Lehner的帮助下,我们想出了如何获得小数点,但我不能使用我已经创建的公式来计算里程数。我必须能够根据radio_name搜索long / lat值,以便我可以单独查看每个车辆的里程数,这就是我创建一个名为CTE和CTE2的临时表的原因。但是,如果我调用临时表,代码将无效,因为它表示它是无效对象。现在代码工作正常,如果我插入“AVL”(原始表的名称)而不是在我上面的最后一个语句中使用CTE2。但是,不使用新的lat / long值计算里程数。

所以,作为回顾:列是纬度,经度。我们需要将这些列转换为在纬度中的2个字符之后以及在经度中的3个字符之后放置小数点。然后我们需要一个名为Miles的列来计算给定的纬度和经度值之间的里程数。

任何提示,帮助,指示等都非常感谢!谢谢!

USE [system]
GO
With CTE as 
(SELECT * FROM AVL 
WHERE (DATE_TIME between '01/30/2013 00:00:00' AND 
'01/30/2013 23:59:59') AND radio_name = 'MAS7'),
CTE2 as 
(select *,row_number() over(partition by Vehicle_ID order by Date_Time) as RN
FROM CTE)

SELECT *, sqrt((69.1*(previous.Latitude - next.Latitude))*
     (69.1*(previous.Latitude-next.Latitude)) +
     (69.1*(previous.Longitude-next.Longitude)) *
     cos(next.Latitude/57.3) * (69.1*(previous.longitude-next.Longitude)) *
     cos(next.Latitude/57.3)) as Miles
From CTE2 as Previous
Join CTE2 as Next
On previous.Vehicle_ID = Next.Vehicle_ID
AND Previous.RN = 
Next.RN - 1
select cast(Latitude / 1000000.0 as decimal(10, 6)) as Latitude
  , cast(Longitude / 1000000.0 as decimal(10, 6)) as Longitude
from cte2

****编辑****
期望的结果看起来与此类似:

Latitude Longitude  Mileage Radio_Name
36.158500 115.949833 3.444  MAS7
36.340000 115.914667 3.443  MAS7
36.153488 115.944875 4.544  MAS7

(这不是正确的里程数,但它只是我正在寻找的结果的一个例子。)

1 个答案:

答案 0 :(得分:2)

请注意the difference between tamp tables, table variables and CTEs

您当前正在使用Common Table Expressions或CTE,它们更类似于derived tables,因为它们只能用于直接跟随它们的一个选择(或插入,更新等)语句(尽管你可以为声明声明多个CTE,就像你所做的那样)。

因此,当您进入第二个选择语句时,您的CTE已超出范围,您不能再引用它们(或者您必须再次声明它们)。在这种情况下,您可能希望使用table variabletemp table

您可以通过转换CTE定义轻松创建临时表,如下所示:

-- Make a temp table
SELECT *
INTO #TempTableName -- Create a new temp table
FROM AVL 
WHERE DATE_TIME between '01/30/2013 00:00:00' AND '01/30/2013 23:59:59'
    AND radio_name = 'MAS7'

-- Check out our new temp table
SELECT * FROM #TempTableName

-- Clean up our temp table, though this will automatically happen once the connection is dropped
DROP TABLE #TempTableName

此外,在使用临时表时,我会经常检查并在我的触发器的开头和结尾放下它们,如下所示:

if object_id('tempdb..#TempTableName') is not null begin
    drop table #TempTableName
end

如果您的代码中有这些内容,请告知我们您是否在执行实际距离计算时遇到困难。