泰医赵伟 發表於 2020-5-28 13:58:00

C# 数据操作系列 - 18 让Dapper更强的插件

<h1 id="0-前言">0. 前言</h1>
<p>在前一篇中我们讲到了Dapper的应用,但是给我们的感觉Dapper不像个ORM更像一个IDbConnection的扩展。是的,没错。在实际开发中我们经常用Dapper作为对EF Core的补充。当然了Dapper并不仅仅只有这些,就让我们通过这一篇文章去让Dapper更像一个ORM吧。</p>
<h1 id="1-dapper-contrib">1. Dapper Contrib</h1>
<p>DapperContrib 扩展了Dapper对于实体类的CRUD方法:</p>
<p>安装方法:</p>
<p>命令行:</p>
<pre><code class="language-bash">dotnet add package Dapper.Contrib
</code></pre>
<p>NuGet:</p>
<pre><code class="language-powershell">Install-Package Dapper.Contrib
</code></pre>
<p>使用:</p>
<pre><code class="language-c#">using Dapper.Contrib.Extensions;
</code></pre>
<p>这个是一个使得Dapper功能更强大的扩展包,因为支持了CRUD,所以需要对实体类添加配置,该扩展包使用Attribute作为依据进行相关映射配置:</p>
<pre><code class="language-c#">
public class Model
{
   
   
    public int Id{get;set;}
   
    public int Count {get;set;}
   
    public String Name{get;set;}
}
</code></pre>
<p>这是所有的配置,Table用来声明是一个表,必须指定表名,Key表示该属性是数据库主键,ExplicitKey表示这个属性是数据库中显示设置的主键,Computed表示该字段是一个计算字段,Write表示该字段可以设置值进去。需要注意的是: Key和ExplicitKey这两个不能同时标注在一个属性上。</p>
<p>那么接下来,我们看看它扩展了哪些方法:</p>
<p>插入单个对象:</p>
<pre><code class="language-c#">public static long Insert&lt;T&gt;(this IDbConnection connection, T entityToInsert, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
</code></pre>
<p>其中 transcation表示事务,如果指定事务,数据的提交将由事务控制,该方法会返回插入对象的主键(如果对象主键是数字类型)或者返回一个待插入列表中已插入的行数。</p>
<p>获取单个对象:</p>
<pre><code class="language-c#">public static T Get&lt;T&gt;(this IDbConnection connection, dynamic id, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
</code></pre>
<p>通过传入主键,获取一个数据</p>
<p>获取所有数据:</p>
<pre><code class="language-c#">public static IEnumerable&lt;T&gt; GetAll&lt;T&gt;(this IDbConnection connection, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
</code></pre>
<p>更新数据:</p>
<p>Dapper Contrib 提供了一个用来更新的方法:</p>
<pre><code class="language-c#">public static bool Update&lt;T&gt;(this IDbConnection connection, T entityToUpdate, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
</code></pre>
<p>这个方法比较有意思的是</p>
<pre><code class="language-c#">var entity = connection.Get&lt;Model&gt;(1);
entity.Name = "测试1";
connection.Update(entity);
</code></pre>
<p>和</p>
<pre><code class="language-c#">var models = connection.GetAll&lt;Model&gt;();
foreach(var m in models)
{
    Console.WriteLine(m);
    m.StringLength ++;
}
connection.Update(models.AsList());
</code></pre>
<p>都可以,并不会报错。</p>
<p><strong>不过需要注意的是,如果需要更新的实例没有指定主键值(主减属性没有赋值),则不会有任何行发生更新。而且在更新的时候,会更新所有列,不会因为不赋值就不更新。</strong></p>
<p>删除方法有两个:</p>
<pre><code class="language-c#">public static bool Delete&lt;T&gt;(this IDbConnection connection, T entityToDelete, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
public static bool DeleteAll&lt;T&gt;(this IDbConnection connection, IDbTransaction transaction = null, int? commandTimeout = null) where T : class;
</code></pre>
<p>删除也是传入一个实体类,一样也只是需要主键有值,如果没有找到主键对应的数据,则不会有任何变化。Delete与Update一样,如果传入一个List集合也是可以的。</p>
<h1 id="2-dapper-transaction">2. Dapper Transaction</h1>
<p>这个包扩展了Dapper的事务处理能力。虽然是Dapper的扩展包,但是是给IConnection添加了一个扩展方法。使用示例如下:</p>
<pre><code class="language-bash">dotnet add package Dapper.Transaction
</code></pre>
<p>老规矩,记得先把包加进来。</p>
<p>然后代码是这样的:</p>
<pre><code class="language-c#">using Dapper.Transaction;
</code></pre>
<pre><code class="language-c#">using(var connection = new SqliteConnection("Data Source=./demo.db"))
{
    connection.Open();
    var transcation = connection.BeginTransaction();
    // 编写业务代码
    transcation.Commit();
}
</code></pre>
<p>如果使用Dapper Transaction,需要先调用 connection.Open()来确保连接是开启状态。</p>
<p>transcation这个对象可以当做普通的DbTranscation对象,传给Dapper的方法来使用,也可以当做一个开启了事务的Dapper客户端来使用。也就是说,Dapper对IDbConnection扩展的方法,在这个包对IDbTranscation也扩展了响应的方法:</p>
<p><img src="https://img2020.cnblogs.com/other/1266612/202005/1266612-20200528135750318-1255034516.png"></p>
<h1 id="3-dapper-plus">3. Dapper Plus</h1>
<p>这个插件是Dapper上用来处理巨量数据的插件,<strong>但这是个收费版的插件</strong>,不过每个月都有一定的试用期限。想试试的可以下一下:</p>
<pre><code class="language-c#">dotnet add package Z.Dapper.Plus
</code></pre>
<p>使用:</p>
<pre><code class="language-c#">using Z.Dapper.Plus;
</code></pre>
<p>这个插件在使用之前需要先配置实体类与数据库之间的映射关系:</p>
<pre><code class="language-c#">DapperPlusManager.Entity&lt;Customer&gt;().Table("Customers");
DapperPlusManager.Entity&lt;Supplier&gt;().Table("Suppliers").Identity(x =&gt; x.SupplierID);
</code></pre>
<p>该插件支持四组大批量处理方式:</p>
<ul>
<li>Bulk Insert</li>
<li>Bulk Update</li>
<li>Bulk Merge</li>
<li>Bulk Delete</li>
</ul>
<pre><code class="language-c#">// STEP MAPPING
DapperPlusManager.Entity&lt;Supplier&gt;().Table("Suppliers").Identity(x =&gt; x.SupplierID);
DapperPlusManager.Entity&lt;Product&gt;().Table("Products").Identity(x =&gt; x.ProductID);

// STEP BULKINSERT
using (var connection = new SqlConnection(FiddleHelper.GetConnectionStringSqlServerW3Schools()))
{
    connection.BulkInsert(suppliers).ThenForEach(x =&gt; x.Products.ForEach(y =&gt; y.SupplierID =x.SupplierID)).ThenBulkInsert(x =&gt; x.Products);
}

// STEP BULKUPDATE
using (var connection = new SqlConnection(FiddleHelper.GetConnectionStringSqlServerW3Schools()))
{
    connection.BulkUpdate(suppliers, x =&gt; x.Products);
}

// STEP BULKMERGE
using (var connection = new SqlConnection(FiddleHelper.GetConnectionStringSqlServerW3Schools()))
{
    connection.BulkMerge(suppliers).ThenForEach(x =&gt; x.Products.ForEach(y =&gt; y.SupplierID =x.SupplierID)).ThenBulkMerge(x =&gt; x.Products);
}

// STEP BULKDELETE
using (var connection = new SqlConnection(FiddleHelper.GetConnectionStringSqlServerW3Schools()))
{
    connection.BulkDelete(suppliers.SelectMany(x =&gt; x.Products)).BulkDelete(suppliers);
}
</code></pre>
<h1 id="4-总结">4. 总结</h1>
<p>这些插件让Dapper更强,也更具备一个完整的ORM的方法,当然实际开发中需要结合实际需求使用。可能并不是所有的都合适。</p>
<p>Dapper的内容就到此为止了。本来预计下一篇开始 asp.net core的内容,不过有个小伙伴推荐了FreeSql,我看了下感觉挺不错的,就给小伙伴们介绍一下~这一个介绍完成之后,就进入了我期待已久的asp.net core系列了。</p>
<blockquote>
<p>更多内容烦请关注我的博客《高先生小屋》</p>
</blockquote>
<p><img src="https://img2020.cnblogs.com/other/1266612/202005/1266612-20200528135750631-1010381827.png"></p><br><br>
来源:https://www.cnblogs.com/c7jie/p/12980416.html
頁: [1]
查看完整版本: C# 数据操作系列 - 18 让Dapper更强的插件