C# 數據操作系列 - 15 SqlSugar 增刪改查詳解

来源:https://www.cnblogs.com/c7jie/archive/2020/05/24/12952759.html
-Advertisement-
Play Games

0. 前言 繼上一篇,以及上上篇,我們對SqlSugar有了一個大概的認識,但是這並不完美,因為那些都是理論知識,無法描述我們工程開發中實際情況。而這一篇,將帶領小伙伴們一起試著寫一個能在工程中使用的模板類。 1. 創建一個Client SqlSugar在操作的時候需要一個Client,用來管理數據 ...


0. 前言

繼上一篇,以及上上篇,我們對SqlSugar有了一個大概的認識,但是這並不完美,因為那些都是理論知識,無法描述我們工程開發中實際情況。而這一篇,將帶領小伙伴們一起試著寫一個能在工程中使用的模板類。

1. 創建一個Client

SqlSugar在操作的時候需要一個Client,用來管理資料庫連接,並操作資料庫。所以我們寫一個DbContext用來創建Client:

public class DefaultContext
{
    public SqlSugarClient Client { get; }

    public DefaultContext(string connectionString, DbType dbType)
    {
        Client = new SqlSugarClient(new ConnectionConfig
		{
            ConnectionString = connectionString,//"Data Source=./demo.db",
            DbType = dbType,
            IsAutoCloseConnection = true,
            InitKeyType = InitKeyType.Attribute
        });
        Client.CodeFirst.InitTables(typeof(Dept), typeof(Person), typeof(Employee));
        Client.Aop.OnLogExecuting = (sql, paramters) =>
        {
            Console.WriteLine(sql);
        };
    }

    public SimpleClient<T> CreateClient<T>() where T : class, new()
    {
        return Client.GetSimpleClient<T>();
    }
}

SqlSugar 提供了一個SimpleClient,這裡面有很多可以直接拿來用的方法,而且這個是一個泛型類。也就是說我們可以使用它對單個實體類進行操作,這在開發中很重要。

2. 插入數據

對於一個程式而言,數據就像是血液一樣重要。對於ORM框架,插入是一切來源的基礎。所以我們先來看看SqlSugar的插入是怎樣的吧:

2.1 簡單的插入模式

public bool Insert(T insertObj);
public bool InsertRange(T[] insertObjs);
public bool InsertRange(List<T> insertObjs);

這是SqlSugar在SimpleClient里提供的兩個預設插入方法,一個是插入單個實體對象,一個是插入一組對象。

預設情況下,SqlSugar插入並不會將主鍵返回給數據。如果後續操作需要當前數據的主鍵,則可以調用另外一個方法:

public int InsertReturnIdentity(T insertObj);

通過這個方法可以獲取一個預設的int類型主鍵值。

2.2 高級玩法

SqlSugar還有一種插入模式,通過AsInsertable返回一個 IInsertable泛型介面:

public IInsertable<T> AsInsertable(T insertObj);
public IInsertable<T> AsInsertable(T[] insertObjs);
public IInsertable<T> AsInsertable(List<T> insertObjs);

這種模式與SimpleClient的普通插入模式不同,它並不會直接執行插入動作,需要手動調用並執行插入動作:

int ExecuteCommand();

執行動作,然後返回受影響的行數。

bool ExecuteCommandIdentityIntoEntity();

執行動作,然後將主鍵插入實體對象,返回插入結果。執行完成後,主鍵數據保存到實體示例中。

long ExecuteReturnBigIdentity();
int ExecuteReturnIdentity();

執行動作,然後返回主鍵值,不會更新實體。

有一點值得特別註意:

所有會返回主鍵的插入都只針對單個數據,如果一次插入多個數據,並不會返回主鍵信息也無法將主鍵信息更新入實體中

以上都是全列插入,SqlSugar還提供了只插入部分列和忽略某些列兩種模式:

IInsertable<T> InsertColumns(Expression<Func<T, object>> columns);// 滿足條件的插入,其他列則不插入
IInsertable<T> InsertColumns(params string[] columns);//插入指定列名
IInsertable<T> IgnoreColumns(Expression<Func<T, object>> columns);// 忽略滿足條件的列,插入其他列
IInsertable<T> IgnoreColumns(params string[] columns);// 忽略這幾個列
IInsertable<T> IgnoreColumns(bool ignoreNullColumn, bool isOffIdentity = false);//指定是否忽略Null列,並是否強制插入主鍵

3. 更新或插入

介紹完插入,那麼來介紹一下更新。正所謂,沒有更新數據就是一灘死水,有了更新數據才有了變化。所以,就讓我們來看看如何優雅的更新數據吧:

3.1 簡單模式

先來兩個最簡單的:

public bool Update(T updateObj);
public bool UpdateRange(T[] updateObjs);
public bool UpdateRange(List<T> updateObjs);

傳入實體,直接更新到資料庫中,需要註意的是這種更新模式只需要保證主鍵有值,且與之對應即可。

public bool Update(Expression<Func<T, T>> columns, Expression<Func<T, bool>> whereExpression);

這是另一種條件更新,會更新滿足whereExpression的所有元素,更新示例:

personClient.Update(p=>new Person
{
    Age = 1
}, p=>p.Id == 1);

columns需要返回一個要更新的對象的屬性列,也就是在columns中設置需要更新的內容。

3.2 高級模式

同樣,通過AsUpdateable開啟高級模式:

public IUpdateable<T> AsUpdateable(T[] updateObjs);
public IUpdateable<T> AsUpdateable(T updateObj);
public IUpdateable<T> AsUpdateable(List<T> updateObjs);

然後可以針對這些今天更多的操作:

int ExecuteCommand();

返回命令執行影響的行數

bool ExecuteCommandHasChange();

返回是否有變化,也就是影響行數是否大於0。

  • 只更新某些列:
IUpdateable<T> SetColumns(Expression<Func<T, bool>> columns);

更新示例:

personClient.AsUpdateable(d).SetColumns(t=>t.Age ==2).ExecuteCommand();

傳入一個lambda表達式,使數據滿足lambda表達式。要求lambda表達式只能用 == 來判斷列是否等於某個值。

IUpdateable<T> SetColumns(Expression<Func<T, T>> columns);
IUpdateable<T> UpdateColumns(params string[] columns);
IUpdateable<T> UpdateColumns(Expression<Func<T, object>> columns);

傳入要更新的實際列名。其中 object 用來接一個匿名對象,其中屬性名字就是要更新的值。

  • 不更新某些列
IUpdateable<T> IgnoreColumns(params string[] columns);// 忽略傳入的列名
IUpdateable<T> IgnoreColumns(Expression<Func<T, object>> columns);// 用匿名對象表示要忽略的列名
IUpdateable<T> IgnoreColumns(bool ignoreAllNullColumns, bool isOffIdentity = false, bool ignoreAllDefaultValue = false);// 設置是否忽略Null列,是否強制更新主鍵,是否忽略所有預設值列
  • 條件更新
IUpdateable<T> Where(Expression<Func<T, bool>> expression);
IUpdateable<T> Where(string fieldName, string conditionalType, object fieldValue);
IUpdateable<T> Where(string whereSql, object parameters = null);
IUpdateable<T> WhereColumns(Expression<Func<T, object>> columns);
IUpdateable<T> WhereColumns(string columnName);
IUpdateable<T> WhereColumns(string[] columnNames);

來,簡單猜一猜這幾個是什麼意思呢?

可以說很簡單明瞭的幾種條件設置模式,lambda表示篩選更新數據,欄位值判斷條件更新。

其中 conditionType的值,推薦使用 ConditionalType枚舉的值。

3.3 更新或插入

在實際開發中可能會遇到插入或更新是走的一個方法,所以我們就要尋找一個可以直接更新或插入的方法。SqlSugar為此提供瞭解決方案:

ISaveable<T> Saveable<T>(T saveObject) where T : class, new();
ISaveable<T> Saveable<T>(List<T> saveObjects) where T : class, new();

不過這個方法是在SugarClient里,我們可以通過:

public ISqlSugarClient AsSugarClient();

在SimpleClient中獲得 與之關聯的SugarClient對象。

關於更新或插入判斷標準是,主鍵是否有值。如果主鍵有值且在資料庫中存在該條記錄,則執行更新,否則執行插入。

4. 刪除

刪除在實際開發過程中是一個非常重要的功能點,所以如何快速有效的刪除數據也是一件很重要的事。那麼,就來看看如何執行刪除吧:

public bool Delete(Expression<Func<T, bool>> whereExpression);
public bool Delete(T deleteObj);
public bool DeleteById([Dynamic] dynamic id);
public bool DeleteByIds([Dynamic(new[] { false, true })] dynamic[] ids);

刪除沒有其他需要註意的地方,第一個是條件刪除,所有滿足條件的都要刪除。第二個刪除單個對象,後面兩個根據主鍵刪除對象。

悄悄吐槽一下,主鍵的地方用object會比較好一點,因為動態對象會增加一次裝箱拆箱的過程。

當然了,刪除也有AsDeleteable方法。IDeleteable介面特別提供了根據sql語句刪除的方法,除此之外沒有別的需要註意的地方了。

5. 查詢

一個好的ORM框架,至少五分功力在查詢上,如何更快更準的查詢成為了現在開發對ORM框架的要求。同時簡單易用更是程式員對ORM的期望。

那麼我們來看看SqlSugar在查詢上的功力吧:

public bool IsAny(Expression<Func<T, bool>> whereExpression);// 查詢是否存在符合條件的數據
public int Count(Expression<Func<T, bool>> whereExpression);// 獲取滿足條件的數量
public T GetById([Dynamic] dynamic id);//根據主鍵獲取一個實例
public bool IsAny(Expression<Func<T, bool>> whereExpression);//返回滿足條件的一個對象
public List<T> GetList();// 以List的形式返回所有數據
public List<T> GetList(Expression<Func<T, bool>> whereExpression);//返回符合條件的所有數據

分頁獲取數據:

public List<T> GetPageList(Expression<Func<T, bool>> whereExpression, PageModel page);
public List<T> GetPageList(Expression<Func<T, bool>> whereExpression, PageModel page, Expression<Func<T, object>> orderByExpression = null, OrderByType orderByType = OrderByType.Asc);
public List<T> GetPageList(List<IConditionalModel> conditionalList, PageModel page);
public List<T> GetPageList(List<IConditionalModel> conditionalList, PageModel page, Expression<Func<T, object>> orderByExpression = null, OrderByType orderByType = OrderByType.Asc);

其中IConditionModel是一個空的介面,用來定義規範查詢規範,實際上使用的是類:

public class ConditionalModel: IConditionalModel
{
    public ConditionalModel()
    {
        this.ConditionalType = ConditionalType.Equal;
    }
    public string FieldName { get; set; }
    public string FieldValue { get; set; }
    public ConditionalType ConditionalType { get; set; }
    public Func<string,object> FieldValueConvertFunc { get; set; }
}

那麼,我們看一下 ConditionType,定義了各種判斷依據:

public enum ConditionalType
{
    Equal=0,
    Like=1,
    GreaterThan =2,
    GreaterThanOrEqual = 3,
    LessThan=4,
    LessThanOrEqual = 5,
    In=6,
    NotIn=7,
    LikeLeft=8,
    LikeRight=9,
    NoEqual=10,
    IsNullOrEmpty=11,
    IsNot=12,
    NoLike = 13,
}

那麼我們簡單看一下 使用IConditionModel進行分頁是怎樣的效果:

var list = personClient.GetPageList(new List<IConditionalModel>
{
    new ConditionalModel
    {
        FieldName = "Age",
        FieldValue = "3",
        ConditionalType = ConditionalType.LessThan
    }
}, pageModel);

生成如下SQL語句:

SELECT COUNT(1) FROM (SELECT `Id`,`Name`,`Age` FROM `Person`  WHERE   Age < @ConditionalAge0  ) CountTable 
SELECT `Id`,`Name`,`Age` FROM `Person`   WHERE   Age < @ConditionalAge0      LIMIT 0,2

可以看出兩者並沒有區別,只不過是不同的查詢習慣。

6. 總結

按照之前的習慣,到目前應該可以結束了。但是SqlSugar還有一些很重要的地方沒有介紹,所以就加個下期預告

下一篇將為大家分析SqlSugar的一些更高級的內容,查詢的高級模式、事務以及批量操作

好,總結一下這一篇,我們在這一篇看到了SqlSugar在增刪改查上的亮點,可以說更貼合實際業務需求開發。嗯,悄悄給個贊。

再有三篇的內容《C# 數據操作系列》就要完結了。從下一系列開始,就要步入工作中最重要的技術棧了:Asp.net Core。這是可以寫入簡歷的。嗯,沒錯。下一系列計劃以實戰的形式介紹asp.net core的知識點和設置。

更多內容煩請關註我的博客《高先生小屋》

file


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

-Advertisement-
Play Games
更多相關文章
  • IDEA一些不錯的插件分享 目錄 IDEA一些不錯的插件分享 插件集合 CamelCase Translation LiveEdit MarkDown Navigator Jrebel CheckStyle IDEA Alibaba Java Coding Guidelines Ideavim Ma ...
  • hashCode() 和equals() 方法的重要性體現在什麼地方? Java中的HashMap使用hashCode()和equals()方法設置值,根據鍵獲取值的時候也會用到這兩個方法。 怎樣 設置 的值? hashCode()獲得 hash值。而hash值用來確定hashmap中內部 Node ...
  • 老孟導讀:今天分享一個類似“孔雀開屏”的動畫效果,打開新的頁面時,新的頁面從屏幕右上角以圓形逐漸打開到全屏。 先來看下具體的效果 不知道這種效果大家叫什麼名字?如果有更合適的名字可以在評論處告訴我,下麵來說下如何實現此效果。 在使用Navigator進入一個新的頁面時,通常用法如下: 就包含了切換頁 ...
  • 4.元組 元組的主要特性為: 1.元組在創建之後,具有不可以更改的特性,因此不能直接給元組的元素賦值 2.元組的元素類型可以為任意類型,如字典、字元串、列表等 3.元組常用於在程式的整個生命周期中都不變的場景中 4.1 常用方法 元組大小和內容在定義賦值之後,就不可更改,常用的方法如下所示: cou ...
  • 1.概念簡述 (1)AR模型 AR 模型(auto regressive model)自回歸模型,模型參量法高解析度譜分析方法之一,也是現代譜估計中常用的模型。 用AR模型法求信具體作法是: ①選擇AR模型,在輸入是衝激函數或白雜訊的情況下,使其輸出等於所研究的信號,至少,應是對該信號的一個好的近似 ...
  • 執行代碼清理時,可以點擊那個掃把小圖片,會按照預設的第一種配置文件來自動修複。也可以點擊下拉三角符合,選擇不同的配置文件,然後進行修複。或者快捷鍵Ctrl+K,Ctrl+E。 針對每一項配置的說明: 刪除不必要的using 儘可能將私有欄位設置為只讀 刪除不必要的類型轉換(針對強類型轉換),像Con ...
  • .NET 程式下銳浪報表 (Grid++ Report) 的綠色發佈指南 在銳浪報表官方為 CSharp 編寫的開發文檔:“在C#與VB.NET中開始使用說明.txt” 中,關於發佈項目是這麼描述的: ★發佈你的項目,用VS.NET製作安裝程式:1、先創建安裝項目:在解決方案資源管理器的根節點上點右 ...
  • 1 class Program 2 { 3 static void Main(string[] args) 4 { 5 //數組:長度不可變,類型單一 6 //ArrayList集合:長度可以任意改變,類型可以不單一 7 8 //創建一個ArrayList對象 9 ArrayList mylist ...
一周排行
    -Advertisement-
    Play Games
  • Timer是什麼 Timer 是一種用於創建定期粒度行為的機制。 與標準的 .NET System.Threading.Timer 類相似,Orleans 的 Timer 允許在一段時間後執行特定的操作,或者在特定的時間間隔內重覆執行操作。 它在分散式系統中具有重要作用,特別是在處理需要周期性執行的 ...
  • 前言 相信很多做WPF開發的小伙伴都遇到過表格類的需求,雖然現有的Grid控制項也能實現,但是使用起來的體驗感並不好,比如要實現一個Excel中的表格效果,估計你能想到的第一個方法就是套Border控制項,用這種方法你需要控制每個Border的邊框,並且在一堆Bordr中找到Grid.Row,Grid. ...
  • .NET C#程式啟動閃退,目錄導致的問題 這是第2次踩這個坑了,很小的編程細節,容易忽略,所以寫個博客,分享給大家。 1.第一次坑:是windows 系統把程式運行成服務,找不到配置文件,原因是以服務運行它的工作目錄是在C:\Windows\System32 2.本次坑:WPF桌面程式通過註冊表設 ...
  • 在分散式系統中,數據的持久化是至關重要的一環。 Orleans 7 引入了強大的持久化功能,使得在分散式環境下管理數據變得更加輕鬆和可靠。 本文將介紹什麼是 Orleans 7 的持久化,如何設置它以及相應的代碼示例。 什麼是 Orleans 7 的持久化? Orleans 7 的持久化是指將 Or ...
  • 前言 .NET Feature Management 是一個用於管理應用程式功能的庫,它可以幫助開發人員在應用程式中輕鬆地添加、移除和管理功能。使用 Feature Management,開發人員可以根據不同用戶、環境或其他條件來動態地控制應用程式中的功能。這使得開發人員可以更靈活地管理應用程式的功 ...
  • 在 WPF 應用程式中,拖放操作是實現用戶交互的重要組成部分。通過拖放操作,用戶可以輕鬆地將數據從一個位置移動到另一個位置,或者將控制項從一個容器移動到另一個容器。然而,WPF 中預設的拖放操作可能並不是那麼好用。為瞭解決這個問題,我們可以自定義一個 Panel 來實現更簡單的拖拽操作。 自定義 Pa ...
  • 在實際使用中,由於涉及到不同編程語言之間互相調用,導致C++ 中的OpenCV與C#中的OpenCvSharp 圖像數據在不同編程語言之間難以有效傳遞。在本文中我們將結合OpenCvSharp源碼實現原理,探究兩種數據之間的通信方式。 ...
  • 一、前言 這是一篇搭建許可權管理系統的系列文章。 隨著網路的發展,信息安全對應任何企業來說都越發的重要,而本系列文章將和大家一起一步一步搭建一個全新的許可權管理系統。 說明:由於搭建一個全新的項目過於繁瑣,所有作者將挑選核心代碼和核心思路進行分享。 二、技術選擇 三、開始設計 1、自主搭建vue前端和. ...
  • Csharper中的表達式樹 這節課來瞭解一下表示式樹是什麼? 在C#中,表達式樹是一種數據結構,它可以表示一些代碼塊,如Lambda表達式或查詢表達式。表達式樹使你能夠查看和操作數據,就像你可以查看和操作代碼一樣。它們通常用於創建動態查詢和解析表達式。 一、認識表達式樹 為什麼要這樣說?它和委托有 ...
  • 在使用Django等框架來操作MySQL時,實際上底層還是通過Python來操作的,首先需要安裝一個驅動程式,在Python3中,驅動程式有多種選擇,比如有pymysql以及mysqlclient等。使用pip命令安裝mysqlclient失敗應如何解決? 安裝的python版本說明 機器同時安裝了 ...