C# 數(shù)據(jù)操作系列 - 15 SqlSugar 增刪改查詳解

0. 前言

繼上一篇聪全,以及上上篇,我們對SqlSugar有了一個(gè)大概的認(rèn)識辅辩,但是這并不完美荔烧,因?yàn)槟切┒际抢碚撝R吱七,無法描述我們工程開發(fā)中實(shí)際情況。而這一篇鹤竭,將帶領(lǐng)小伙伴們一起試著寫一個(gè)能在工程中使用的模板類。

1. 創(chuàng)建一個(gè)Client

SqlSugar在操作的時(shí)候需要一個(gè)Client景醇,用來管理數(shù)據(jù)庫連接臀稚,并操作數(shù)據(jù)庫。所以我們寫一個(gè)DbContext用來創(chuàng)建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 提供了一個(gè)SimpleClient三痰,這里面有很多可以直接拿來用的方法吧寺,而且這個(gè)是一個(gè)泛型類。也就是說我們可以使用它對單個(gè)實(shí)體類進(jìn)行操作散劫,這在開發(fā)中很重要稚机。

2. 插入數(shù)據(jù)

對于一個(gè)程序而言,數(shù)據(jù)就像是血液一樣重要获搏。對于ORM框架赖条,插入是一切來源的基礎(chǔ)。所以我們先來看看SqlSugar的插入是怎樣的吧:

2.1 簡單的插入模式

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

這是SqlSugar在SimpleClient里提供的兩個(gè)默認(rèn)插入方法常熙,一個(gè)是插入單個(gè)實(shí)體對象纬乍,一個(gè)是插入一組對象。

默認(rèn)情況下裸卫,SqlSugar插入并不會將主鍵返回給數(shù)據(jù)仿贬。如果后續(xù)操作需要當(dāng)前數(shù)據(jù)的主鍵,則可以調(diào)用另外一個(gè)方法:

public int InsertReturnIdentity(T insertObj);

通過這個(gè)方法可以獲取一個(gè)默認(rèn)的int類型主鍵值墓贿。

2.2 高級玩法

SqlSugar還有一種插入模式茧泪,通過AsInsertable返回一個(gè) IInsertable泛型接口:

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

這種模式與SimpleClient的普通插入模式不同,它并不會直接執(zhí)行插入動作聋袋,需要手動調(diào)用并執(zhí)行插入動作:

int ExecuteCommand();

執(zhí)行動作队伟,然后返回受影響的行數(shù)。

bool ExecuteCommandIdentityIntoEntity();

執(zhí)行動作舱馅,然后將主鍵插入實(shí)體對象缰泡,返回插入結(jié)果。執(zhí)行完成后代嗤,主鍵數(shù)據(jù)保存到實(shí)體示例中棘钞。

long ExecuteReturnBigIdentity();
int ExecuteReturnIdentity();

執(zhí)行動作,然后返回主鍵值干毅,不會更新實(shí)體宜猜。

有一點(diǎn)值得特別注意:

所有會返回主鍵的插入都只針對單個(gè)數(shù)據(jù),如果一次插入多個(gè)數(shù)據(jù)硝逢,并不會返回主鍵信息也無法將主鍵信息更新入實(shí)體中姨拥。

以上都是全列插入绅喉,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);// 忽略這幾個(gè)列
IInsertable<T> IgnoreColumns(bool ignoreNullColumn, bool isOffIdentity = false);//指定是否忽略Null列柴罐,并是否強(qiáng)制插入主鍵

3. 更新或插入

介紹完插入,那么來介紹一下更新憨奸。正所謂革屠,沒有更新數(shù)據(jù)就是一灘死水,有了更新數(shù)據(jù)才有了變化排宰。所以似芝,就讓我們來看看如何優(yōu)雅的更新數(shù)據(jù)吧:

3.1 簡單模式

先來兩個(gè)最簡單的:

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

傳入實(shí)體,直接更新到數(shù)據(jù)庫中板甘,需要注意的是這種更新模式只需要保證主鍵有值党瓮,且與之對應(yīng)即可。

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需要返回一個(gè)要更新的對象的屬性列,也就是在columns中設(shè)置需要更新的內(nèi)容傲醉。

3.2 高級模式

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

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

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

int ExecuteCommand();

返回命令執(zhí)行影響的行數(shù)

bool ExecuteCommandHasChange();

返回是否有變化,也就是影響行數(shù)是否大于0硬毕。

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

更新示例:

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

傳入一個(gè)lambda表達(dá)式呻引,使數(shù)據(jù)滿足lambda表達(dá)式。要求lambda表達(dá)式只能用 == 來判斷列是否等于某個(gè)值吐咳。

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

傳入要更新的實(shí)際列名逻悠。其中 object 用來接一個(gè)匿名對象,其中屬性名字就是要更新的值韭脊。

  • 不更新某些列
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);// 設(shè)置是否忽略Null列童谒,是否強(qiáng)制更新主鍵,是否忽略所有默認(rèn)值列
  • 條件更新
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);

來沪羔,簡單猜一猜這幾個(gè)是什么意思呢饥伊?

可以說很簡單明了的幾種條件設(shè)置模式,lambda表示篩選更新數(shù)據(jù)蔫饰,字段值判斷條件更新琅豆。

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

3.3 更新或插入

在實(shí)際開發(fā)中可能會遇到插入或更新是走的一個(gè)方法茫因,所以我們就要尋找一個(gè)可以直接更新或插入的方法。SqlSugar為此提供了解決方案:

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

不過這個(gè)方法是在SugarClient里杖剪,我們可以通過:

public ISqlSugarClient AsSugarClient();

在SimpleClient中獲得 與之關(guān)聯(lián)的SugarClient對象冻押。

關(guān)于更新或插入判斷標(biāo)準(zhǔn)是驰贷,主鍵是否有值。如果主鍵有值且在數(shù)據(jù)庫中存在該條記錄洛巢,則執(zhí)行更新括袒,否則執(zhí)行插入。

4. 刪除

刪除在實(shí)際開發(fā)過程中是一個(gè)非常重要的功能點(diǎn)稿茉,所以如何快速有效的刪除數(shù)據(jù)也是一件很重要的事箱熬。那么,就來看看如何執(zhí)行刪除吧:

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);

刪除沒有其他需要注意的地方狈邑,第一個(gè)是條件刪除,所有滿足條件的都要?jiǎng)h除蚤认。第二個(gè)刪除單個(gè)對象米苹,后面兩個(gè)根據(jù)主鍵刪除對象。

悄悄吐槽一下砰琢,主鍵的地方用object會比較好一點(diǎn)蘸嘶,因?yàn)閯討B(tài)對象會增加一次裝箱拆箱的過程。

當(dāng)然了陪汽,刪除也有AsDeleteable方法训唱。IDeleteable接口特別提供了根據(jù)sql語句刪除的方法,除此之外沒有別的需要注意的地方了挚冤。

5. 查詢

一個(gè)好的ORM框架况增,至少五分功力在查詢上,如何更快更準(zhǔn)的查詢成為了現(xiàn)在開發(fā)對ORM框架的要求训挡。同時(shí)簡單易用更是程序員對ORM的期望澳骤。

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

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

分頁獲取數(shù)據(jù):

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是一個(gè)空的接口,用來定義規(guī)范查詢規(guī)范澜薄,實(shí)際上使用的是類:

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,定義了各種判斷依據(jù):

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進(jìn)行分頁是怎樣的效果:

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

可以看出兩者并沒有區(qū)別肤京,只不過是不同的查詢習(xí)慣颊艳。

6. 總結(jié)

按照之前的習(xí)慣,到目前應(yīng)該可以結(jié)束了忘分。但是SqlSugar還有一些很重要的地方?jīng)]有介紹棋枕,所以就加個(gè)下期預(yù)告

下一篇將為大家分析SqlSugar的一些更高級的內(nèi)容,查詢的高級模式饭庞、事務(wù)以及批量操作

好戒悠,總結(jié)一下這一篇,我們在這一篇看到了SqlSugar在增刪改查上的亮點(diǎn)舟山,可以說更貼合實(shí)際業(yè)務(wù)需求開發(fā)绸狐。嗯卤恳,悄悄給個(gè)贊。

再有三篇的內(nèi)容《C# 數(shù)據(jù)操作系列》就要完結(jié)了寒矿。從下一系列開始突琳,就要步入工作中最重要的技術(shù)棧了:Asp.net Core。這是可以寫入簡歷的符相。嗯拆融,沒錯(cuò)。下一系列計(jì)劃以實(shí)戰(zhàn)的形式介紹asp.net core的知識點(diǎn)和設(shè)置啊终。

更多內(nèi)容煩請關(guān)注我的博客《高先生小屋》

file
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末镜豹,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子蓝牲,更是在濱河造成了極大的恐慌趟脂,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,386評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件例衍,死亡現(xiàn)場離奇詭異昔期,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)佛玄,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,142評論 3 394
  • 文/潘曉璐 我一進(jìn)店門硼一,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人梦抢,你說我怎么就攤上這事般贼。” “怎么了惑申?”我有些...
    開封第一講書人閱讀 164,704評論 0 353
  • 文/不壞的土叔 我叫張陵具伍,是天一觀的道長。 經(jīng)常有香客問我圈驼,道長人芽,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,702評論 1 294
  • 正文 為了忘掉前任绩脆,我火速辦了婚禮萤厅,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘靴迫。我一直安慰自己惕味,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,716評論 6 392
  • 文/花漫 我一把揭開白布玉锌。 她就那樣靜靜地躺著名挥,像睡著了一般。 火紅的嫁衣襯著肌膚如雪主守。 梳的紋絲不亂的頭發(fā)上禀倔,一...
    開封第一講書人閱讀 51,573評論 1 305
  • 那天榄融,我揣著相機(jī)與錄音,去河邊找鬼救湖。 笑死愧杯,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的鞋既。 我是一名探鬼主播力九,決...
    沈念sama閱讀 40,314評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼邑闺!你這毒婦竟也來了跌前?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,230評論 0 276
  • 序言:老撾萬榮一對情侶失蹤陡舅,失蹤者是張志新(化名)和其女友劉穎舒萎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蹭沛,經(jīng)...
    沈念sama閱讀 45,680評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,873評論 3 336
  • 正文 我和宋清朗相戀三年章鲤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了摊灭。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,991評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡败徊,死狀恐怖帚呼,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情皱蹦,我是刑警寧澤煤杀,帶...
    沈念sama閱讀 35,706評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站沪哺,受9級特大地震影響沈自,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜辜妓,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,329評論 3 330
  • 文/蒙蒙 一枯途、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧籍滴,春花似錦酪夷、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,910評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至勋功,卻和暖如春坦报,著一層夾襖步出監(jiān)牢的瞬間库说,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,038評論 1 270
  • 我被黑心中介騙來泰國打工燎竖, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留璃弄,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,158評論 3 370
  • 正文 我出身青樓构回,卻偏偏與公主長得像夏块,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子纤掸,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,941評論 2 355

推薦閱讀更多精彩內(nèi)容