[C#]利用 DynamicLinq 實現簡單的動態表達式構建查詢

来源:https://www.cnblogs.com/myzony/archive/2018/06/06/9143692.html
-Advertisement-
Play Games

平時使用 LINQ 進行一些簡單的條件拼接查詢一般都會這樣操作: 這裡有三個條件,是前端傳入的搜索條件,然後我們來編寫一個查詢語句: 因為我們前端傳入的條件不是固定的,所以有可能會出現有的條件沒有傳入的情況,如果是 SQL 的動態拼接 SQL 就可以了,而 Linq 你肯定是沒法動態拼接的,只有自己 ...


平時使用 LINQ 進行一些簡單的條件拼接查詢一般都會這樣操作:

public class SearchInputDto
{
    public string ConditionA { get; set; }
    public int? ConditionB { get; set; }
    public string ConditionC { get; set; }
}

這裡有三個條件,是前端傳入的搜索條件,然後我們來編寫一個查詢語句:

public Task Search(SearchInputDto input)
{
    var queryResult = _db.Where(z=>(input.ConditionA == null || z.Name == input.ConditionA) 
                                && (input.ConditionB == null || z.Number == input.ConditionB)
                                && (input.ConditionC == null || z.Address == input.ConditionC));
    
    // 執行其他操作...
    
    return Task.FromResult(0);
}

因為我們前端傳入的條件不是固定的,所以有可能會出現有的條件沒有傳入的情況,如果是 SQL 的動態拼接 SQL 就可以了,而 Linq 你肯定是沒法動態拼接的,只有自己構建一個表達式樹傳入到 IQuerable<T>.Where(Expression<Func<T,bool>> expression) 裡面進行查詢。

純手工構建表達式樹也不是不可以,只是略微麻煩,而我們則可以藉助 System.Linq.Dynamic.Core 來方便的實現動態查詢語句拼接。

他的常規用法如下:

官方 WIKI 地址:https://github.com/StefH/System.Linq.Dynamic.Core/wiki/Dynamic-Expressions

var query = db.Customers
              .Where("City == @0 and Orders.Count >= @1", "London", 10)
              .OrderBy("CompanyName")
              .Select("new(CompanyName as Name, Phone)");

既然是字元串那麼就可以拼接,我們來做一下改造。

首先去 NuGet 當中搜索 System.Linq.Dynamic.Core 庫,安裝之後我們來重新編寫之前的查詢範例,首先我們來寫一個構建器,用於構建我們的表達式樹:

using Abp.Runtime.Caching;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Dynamic.Core;
using System.Linq.Expressions;
using System.Reflection;
using System.Text;

namespace Abp.Linq.Expressions
{
    public class ExpressionBuilder<TEntity, TSearchDto>
    {
        // 其實這裡也可以通過傳入 params Expression<Func<TRelateEntity, object>>[] selectFields 來構建
        public Expression<Func<TEntity, bool>> Build(string[] excludeFields, TSearchDto dto)
        {
            var parameters = GenerateParametersDictionary(excludeFields, dto);

            StringBuilder sb = new StringBuilder();
            var fieldNames = parameters.Keys.ToList();

            // 動態拼接
            for (int i = 0; i < fieldNames.Count; i++)
            {
                sb.Append(fieldNames[i]).Append($" == @{i}").Append(" && ");
            }

            var lambdaStr = sb.ToString();
            lambdaStr = lambdaStr.Substring(0, lambdaStr.Length - " && ".Length);

            // 構建表達式
            return DynamicExpressionParser.ParseLambda<TEntity, bool>(new ParsingConfig(), false, lambdaStr, parameters.Values.ToArray());
        }

        // 構建參數/值鍵值對,如果參數值為 NULL 則不進行構建
        private Dictionary<string, object> GenerateParametersDictionary(string[] excludeFields, TSearchDto dto)
        {
            var typeInfo = typeof(TSearchDto);
            var properties = typeInfo.GetProperties();
            var parameters = new Dictionary<string, object>();

            foreach (var property in properties)
            {
                var propertyValue = property.GetValue(dto);

                if (propertyValue == null) continue;
                if (excludeFields == null) continue;
                if (excludeFields.Contains(property.Name)) continue;
                if (parameters.ContainsKey(property.Name)) continue;

                parameters.Add(property.Name, propertyValue);
            }

            return parameters;
        }
    }
}

用法很簡單,用剛纔的代碼作為一個例子:

public Task Search(SearchInputDto input)
{
    var builder = new ExpressionBuilder<EntityA,SearchInputDto>();
    var queryResult = _db.Where(builder.Build(null,input));
    
    // 執行其他操作...
    
    return Task.FromResult(0);
}

可以看到已經變得十分簡潔,這裡僅僅作為拋磚引玉,其實還有更多高級的用法,這裡不再贅述。


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

-Advertisement-
Play Games
更多相關文章
  • 本質上適合非同步的操作有:HTTP請求,資料庫指令,Web服務調用等。 1、暫停一段時間(以非同步方式)。 以非同步的方式暫停一段時間,這在進行單元測試或者重試延遲時非常有用。 Task類有一個返回Task對象的靜態函數Delay,下麵是其中的一個 一個簡單的指數退避。指數退避是一種重試策略,重試的延遲時 ...
  • SVN 安裝後右鍵出現點擊滑鼠右鍵彈出錯誤提示:CrashHandler initialization error 原因是目標文件夾中缺少SendRpt.exe文件 解決方案:找svn是好的的同事將bin目錄複製替換本地的bin就可以解決 由於沒有找到上傳附件的地方,就沒有上鄙人的bin目錄了,實屬 ...
  • Scenario: 創建了一個WinForm的小程式,希望將它顯示在任務欄,所以在工具欄中的“公共控制項”里,拖入NotifyIcon控制項—notifyIcon1,這個是程式運行任務欄右側通知區域圖標顯示控制項,為控制項notifyIcon的屬性Icon添加一個icon圖標,或從代碼中加入。 Issue: ...
  • 在一些耗時的操作過程中,在長時間運行時可能會導致用戶界面 (UI) 處於停止響應狀態,用戶在這操作期間無法進行其他的操作,為了不使UI層處於停止響應狀態,我們傾向推薦用戶使用BackgroundWorker來進行處理,這個後臺的線程處理,可以很好的實現常規操作的同時,還可以及時通知UI,包括當前處理... ...
  • Select與Select Many 之前在項目中查詢資料庫中的數據,都是通過sql語句來查詢的,但是隨著時代的發展,微軟在.Net Framework 4.5版中推出的一個主要的特性——LINQ。 LINQ是Language Integrate Query的縮寫,意為語言集成查詢。其中有兩種查詢方 ...
  • Restful幾乎已算是API設計的標準,通過HTTP Method區分新增(Create)、查詢(Read)、修改(Update)和刪除(Delete),簡稱CRUD四種數據存取方式,簡約又直接的風格,讓人用的愛不釋手。本篇將介紹如何通過ASP.NET Core實踐REST-Like API。 為 ...
  • .net 這幾年國內確實不好過。 很多都選擇轉行。不過.net Core跨平臺 開源之後 。社區的生態在慢慢建立。往好的趨勢發展。 對於堅守在.NET戰線的開發者來說 是個挺不錯的消息。 特別是微軟收購75億美金GitHub。.net 生態 社區圈子。肯定會有所上升。 發展趨勢越來越好。(當然 這隻 ...
  • 一、序言 在Office Word中,支持在Word文檔中插入類型非常豐富的形狀,包括線條、矩形、基本形狀(諸如圓形、多邊形、星形、括弧、笑臉等等圖形)、箭頭形狀、公式形狀、流程圖、旗幟圖形、標註圖形等等,我們在編程過程中,想要在Word中繪製不同類型的圖形,可以通過類庫來操作。控制項Spire.Do ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...