C# 数据操作系列 - 15 SqlSugar 增删改查详解(超长篇)
liebian365 2024-10-28 17:16 36 浏览 0 评论
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的知识点和设置。
更多内容烦请关注我的博客《高先生小屋》
相关推荐
- C#委托—马工教你轻松玩转委托(c#中委托)
-
前言大家好,我是马工!在C#中有一个小白谈之色变的知识点叫委托,很多学了一两年C#的还不知道委托究竟是什么东西,本文就来帮你彻底解开解惑,从此委托就是小儿科!...
- AR农业种植指导(农业种植模式)
-
以下是一些常见的AR/VR在农业中的应用场景及相关源码示例的一些思路(注意,完整且复杂的源码通常需要根据具体需求和大量开发工作来完成,这里只是简单示例):1.AR农业种植指导-功能描述:通过AR...
- C#中timer中的enable和start以及stop的作用分别是什么?
-
文章来自AI问答的摘录。在C#中,System.Timers.Timer和System.Windows.Forms.Timer都提供了Enabled属性以及Start()和Stop()方法来控制定时器...
- AOP实现原理浅析(简述aop的原理以及作用)
-
在C#中实现一个简单的动态代理框架,可以通过反射和System.Reflection.Emit命名空间动态生成代理类。以下是一个分步指南和示例代码:...
- 如何防止数据库服务器的 SQL 注入攻击
-
如何防止数据库服务器的SQL注入攻击防止SQL注入攻击需要从代码开发、数据库配置到安全运维的全流程防护。以下是详细且可操作的防护策略和步骤:...
- Java静态三剑客(静态内部类/静态方法/静态类)的正确打开方式
-
作为Java程序员必知的"静态三剑客",它们看似简单却暗藏玄机!本文将带你解锁高阶玩法,避免99%开发者都会踩的坑!一、概念速览(颠覆传统认知)...
- 生成AutoCAD图纸批量合并代码(autocad怎么合并图纸)
-
想要生成AutoCAD图纸批量合并的代码。首先,我需要确定他使用哪个版本的AutoCAD,因为不同版本的API可能不同,比如VBA、.NET(C#或VB.NET)或者AutoLISP。不过用户没有具体...
- C#从入门到精通(4)—Linq实现集合快速转换、筛选、排序、分组
-
前言大家好,我是上位机马工,硕士毕业4年年入40万,目前在一家自动化公司担任软件经理,从事C#上位机软件开发8年以上!我们在C#开发中需要经常对集合中的数据进行处理,比如我需要将一个Int类型的集合数...
- CSharp(dotNet)绝版面试题,面试大全(二)
-
21.简述C#的密封类和密封方法?C#中,密封类和密封方法都是使用sealed修饰。密封类表示这个类不能被继承。密封方法表示这个方法不能被重写,和virtual方法相对。22.请介绍C#静态构造器的特...
- C#从入门到精通(7)—C#裁剪图像的几种方法总结
-
前言大家好,我是上位机马工,硕士毕业4年年入40万,目前在一家自动化公司担任软件经理,从事C#上位机软件开发8年以上!我们在上位机软件开发过程中经常需要裁剪图像,本文就是对c#中常见的裁剪图像方法进行...
- Csharp 进度条的使用(c#进度条是哪个控件)
-
【开发过程】(1)创建一个Windows应用程序,项目名称为Ex047,窗体默认为Form"(2)在窗体中添加菜单、工具栏和状态栏。(3)面添加ProgressBar.设置。(4)在窗体载入Loa...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- wireshark怎么抓包 (75)
- qt sleep (64)
- cs1.6指令代码大全 (55)
- factory-method (60)
- sqlite3_bind_blob (52)
- hibernate update (63)
- c++ base64 (70)
- nc 命令 (52)
- wm_close (51)
- epollin (51)
- sqlca.sqlcode (57)
- lua ipairs (60)
- tv_usec (64)
- 命令行进入文件夹 (53)
- postgresql array (57)
- statfs函数 (57)
- .project文件 (54)
- lua require (56)
- for_each (67)
- c#工厂模式 (57)
- wxsqlite3 (66)
- dmesg -c (58)
- fopen参数 (53)
- tar -zxvf -c (55)
- 速递查询 (52)