前言 一個小需求的實現,做一個備忘,個人理解,可能存在錯誤。 客戶有很多設備,這些設備分散在不同的地方,現在需要通過小程式獲取附近的(比如1000米)之類的設備列表,以距離排序 第一個想到的的是找百度/騰訊等地圖,看有沒有提供相應的開放api,先將我們的設備id以及對應的經緯度存儲到地圖中,再調用某 ...
前言
一個小需求的實現,做一個備忘,個人理解,可能存在錯誤。
客戶有很多設備,這些設備分散在不同的地方,現在需要通過小程式獲取附近的(比如1000米)之類的設備列表,以距離排序
第一個想到的的是找百度/騰訊等地圖,看有沒有提供相應的開放api,先將我們的設備id以及對應的經緯度存儲到地圖中,再調用某個api,傳入我當前坐標經緯度,然後返回附近的設備列表。最後沒找到這個介面。應該是有,但是我沒找到
第二個想能不能通過c#用啥演算法實現,太菜搞不定
最後想到sqlserver有個空間數據的概念,最終勉強實現。
參考:空間數據 (SQL Server) efcore中使用空間數據
概述
空間對象其實就是點、線、面之類的意思,我們希望對其進行一些操作,如:查詢得到兩個面之間的交集/差集/並集;以一個點為中心,設置半徑得到一個面;查詢兩個點之間的距離;等等。這些操作如果我們自己用演算法來實現想想有點怕。
c#提供了相應的庫來表示這些空間對象,也提供了相應的方法來執行針對空間對象的操作 ->NetTopologySuite(從java的JTS移植來的)由於這次小需求只是依賴了資料庫對空間數據的支持,所以沒有詳細研究NetTopologySuite(資料少)
sqlserver2008開始支持空間數據,它也提供了相應的類型來表示空間對象,也提供相應的函數來操作這些對象,當然其它數據也支持
估計還有單獨的空間數據引擎,以文檔形式存儲空間對象
這些點/線/面可以放在一個普通平面中,這時把他們看成幾何對象,比如一個點就可以用XY坐標來表示、一個線就可以用兩個點來表示...;
也可以將這些空間對象放在地球環境中來看,那麼點就對象就用經緯度來表示、線就用兩個點來表示
幾何圖形與地理位置
同一個空間對象,比如一個點 可以把它放在一個普通的平面坐標中來看待,用XY坐標來表示,也可以放進地球環境中用經緯度來表示。
- Geometry 類型表示歐幾裡得(平面)坐標系中的數據。
- Geography 類型表示圓形地球坐標系中的數據。
以點來說,可以創建Geometry類型的點,也可以創建Geography類型的點。資料庫和c#都有對象的創建方式。同理 線、面 都分為這兩種類型
空間數據類型
就是說的上面的點、線、面 當然還有更多類型。NetTopologySuite庫提供了相應的類型來表示。sqlserver中也有相應的類型,但是sqlserver存儲空間對象的欄位的類型只分為Geography和Geometry,只是在插入值的時候會創建不同的對象
空間引用標識符 (SRID)
文檔的說明看不太懂,反正對我們的限制就是 若我們使用Geography類型的空間對象,那麼必須為每個空間對象設置SRID,對多個空間進行操作(比如看距離)時,這些空間對象的SRID必須一樣。
感覺就像是有多個地球,每個地球有一個編號。同一個點(經緯度)在不同的地球(SRID)中其實位置不一樣。不曉得這樣理解對不。
在當前需求中反正寫死4326,文檔中經常用這個。
各種函數
實例方法:一個空間對象就是一個實例。在sql語句中可以實例化一個空間對象,數據表那個存儲空間對象的單元格中存儲的就是一個具體的空間對象實例
靜態方法:不屬於空間對象實例的,通過字元串創建空間對象,這個方法放實例上不合適
其實跟c#中的實例和靜態一個意思
OGC:應該是個什麼標準,定義了空間數據的通用操作,當然分為實例OGC方法和靜態OGC方法
微軟對標準進行了擴展,實例和靜態都有擴展
原始SQL和EFCore實現
efcore不熟 又是第一次用空間數據,所以沒用efcore來實現,本身用的abp 預設是用的efcore,用這個方式本來應該更方便。efcore中使用空間數據
原始sql就簡單了。
沒有用文檔中的空間索引。也沒有直接存儲空間對象,而是查詢時構建
爛尾....