entity framework 實現按照距離排序

来源:https://www.cnblogs.com/hucheng/archive/2018/10/31/9885573.html
-Advertisement-
Play Games

在做項目時,經常會遇到“離我最近”這種需求。顧名思義,它需要根據用戶的經緯度和事物的經緯度計算距離,然後進行排序,最後分頁(當然這些操作要在資料庫中進行,否則就變成假分頁了)。 我們通常可以用sql語句來實現 但是我比較習慣使用 entity framework,於是我就想著能不能用 entity ...


在做項目時,經常會遇到“離我最近”這種需求。顧名思義,它需要根據用戶的經緯度和事物的經緯度計算距離,然後進行排序,最後分頁(當然這些操作要在資料庫中進行,否則就變成假分頁了)。

我們通常可以用sql語句來實現

SELECT
    es_name,
    es_lon,
    es_lat,
    ROUND(
        6378.138 * 2 * ASIN(
            SQRT(
                POW(
                    SIN(
                        (
                            30.611842 * PI() / 180 - es_lat * PI() / 180
                        ) / 2
                    ),
                    2
                ) + COS(30.611842 * PI() / 180) * COS(es_lat * PI() / 180) * POW(
                    SIN(
                        (
                            104.074666 * PI() / 180 - es_lon * PI() / 180
                        ) / 2
                    ),
                    2
                )
            )
        ) * 1000
    ) AS distance_um
FROM
    c_ershuai
ORDER BY
    distance_um ASC

但是我比較習慣使用 entity framework,於是我就想著能不能用 entity framework 實現按照距離排序。

 

以下是我採用的方案

首先定義一個介面,用來表示具有經緯度信息的實體。

    /// <summary>
    /// 具有經緯度
    /// </summary>
    public interface IHasLngAndLat
    {
        /// <summary>
        /// 經度
        /// </summary>
        double Lng { get; set; }
        /// <summary>
        /// 緯度
        /// </summary>
        double Lat { get; set; }
    }

然後創建泛型類,用來包裝計算的距離。

    /// <summary>
    /// 帶距離的數據
    /// </summary>
    /// <typeparam name="TEntity"></typeparam>
    public class DataWithDistance<TEntity>
    {
        /// <summary>
        /// 距離(km)
        /// </summary>
        public double Distance { get; set; }
        /// <summary>
        /// 實體數據
        /// </summary>
        public TEntity Entity { get; set; }
    }

最後編寫根據距離排序的擴展方法

註意:這個方法是採用的 SqlFunctions 類,所以僅支持SqlServer資料庫,如果是其它資料庫,需要將 SqlFunctions 更換成對應的類

    /// <summary>
    /// IQueryable擴展類
    /// </summary>
    public static class QueryableExtension
    {
        /// <summary>
        /// 根據距離排序
        /// </summary>
        /// <typeparam name="TEntity"></typeparam>
        /// <param name="queryable"></param>
        /// <param name="lng">經度</param>
        /// <param name="lat">緯度</param>
        /// <returns></returns>
        public static IQueryable<DataWithDistance<TEntity>> OrderByDistance<TEntity>(this IQueryable<TEntity> queryable, double lng, double lat) where TEntity : class, IHasLngAndLat
        {
            var rtn = from q in queryable
                      let radLat1 = lat * Math.PI / 180.0
                      let radLat2 = q.Lat * Math.PI / 180.0
                      let a = radLat1 - radLat2
                      let b = lng * Math.PI / 180.0 - q.Lng * Math.PI / 180.0
                      let s = 2 * SqlFunctions.Asin(SqlFunctions.SquareRoot(Math.Pow((double)SqlFunctions.Sin(a / 2), 2) +
               SqlFunctions.Cos(radLat1) * SqlFunctions.Cos(radLat2) * Math.Pow((double)SqlFunctions.Sin(b / 2), 2))) * 6378.137
                      let d = Math.Round((double)s * 10000) / 10000
                      orderby d
                      select new DataWithDistance<TEntity> { Entity = q, Distance = d };

            return rtn;
        }
    }

以上就完成了 entity framework 按照距離排序的功能。

 

接下來我們用它來寫一個小小的demo

首先創建一個商店實體類,具有經緯度欄位,實現了  IHasLngAndLat 介面。

    /// <summary>
    /// 商店實體
    /// </summary>
    public class Shop : IHasLngAndLat
    {
        /// <summary>
        /// 主鍵
        /// </summary>
        public int Id { get; set; }
        /// <summary>
        /// 商店名稱
        /// </summary>
        [Required]
        [StringLength(64)]
        public string ShopName { get; set; }
        /// <summary>
        /// 經度
        /// </summary>
        public double Lng { get; set; }
        /// <summary>
        /// 緯度
        /// </summary>
        public double Lat { get; set; }
    }

然後創建EF上下文類

    /// <summary>
    /// EF上下文
    /// </summary>
    public class DemoDbContext : DbContext
    {
        public DemoDbContext()
            : base("name=DemoDbContext")
        {
        }
        public virtual DbSet<Shop> Shop { get; set; }
    }

最後我們分頁查詢商店,並按照距離由近到遠排序

            #region 入參
            double user_lng = 113.46, user_lat = 22.27;  //用戶經緯度
            int pageIndex = 3; //當前頁碼
            int pageSize = 10; //每頁條數
            #endregion

            using (DemoDbContext context = new DemoDbContext())
            {
                var queryable = context.Shop.AsNoTracking().AsQueryable();
                IQueryable<DataWithDistance<Shop>> sort_queryable = queryable.OrderByDistance(user_lng, user_lat);  //按照用戶的距離從近到遠排序
                List<DataWithDistance<Shop>> data = sort_queryable.Skip((pageIndex - 1) * pageSize).Take(pageSize).ToList();   //分頁並執行sql查詢獲取數據

                //TODO:將查到的數據映射成DTO對象,並返回給客戶端
            }

好了,entity framework 實現按照距離排序 也就全部完成了。

 

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • netty 與 webSocket 起因 有個需求需要用到 ,然後最近又正好在學 ,然後合起來走一波。寫篇文章記錄一下,做一個念想。 協議格式 開始 我們先寫一個什麼都不加的 熱熱手,話不多說,代碼如下 常規的netty入門示例,加了個String的編碼和解碼器,還加了一個列印消息的 ,並不是什麼太 ...
  • 1. 文件操作 open() 文件句柄 open()打開一個文件, 獲取的是文件句柄 read() #讀取全部內容 read(n)#讀取前n個字元 readline()#讀取一行 且讀取出來末尾都有\n readlines()#讀取全部 每一⾏形成一個元素並放到列表 註意: 讀取完的文件句柄一定要關 ...
  • package zrs; public class javaDay02_3 { public static void main(String[] args){ //switch 結構 int x=5; switch(x){//x 支持byte short int char 5.0以後 enum St ...
  • ·字元串(string) @ title @ title @ 小結: ·字元串(string) @ title @ title @ 小結: ****** 幾米花的Python ****** 博客主頁:https://www.cnblogs.com/jimmy-share/ 歡迎轉載 ~ ...
  • [TOC] 1. 函數名的使用 其實函數名也是一個變數,但它是一個比較特殊的變數,與小括弧配合可以執行函數的變數: 函數名其實和記憶體一樣,也可以使用 查看它的記憶體地址: 函數名賦值給其他變數 函數也能當作容器類的元素: 函數名也能當作函數的參數: 函數名也可以作為函數的返回值: 2. 閉包 閉包是指 ...
  • 界面如下圖: 動圖效果演示: ...
  • 加群731738614加群731738614 ...
  • 最近項目上需要通過MVVM來控制TreeView,其中需要需要控制通過搜索來定位某個節點,正常邏輯下,首先通過需要在樹上面找到該節點,然後選中該節點,並將該節點的父節點展開,這個時候需要通過MVVM來控制,需要綁定起來,只是一直沒有binding上,代碼如下: MVVM示例代碼: 界面代碼: 數據的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...