Dapper源码学习和源码修改(下篇)

继上篇Dapper源码学习和源码修改 讲了下自己学习Dapper的心得之后,下篇也随之而来,上篇主要讲的入参解析那下篇自然主打出参映射了。

好了,废话不多说,开始吧。

学习之前你的先学习怎么使用Dapper,这个我在上篇都提过,如果没使用过Dapper的同学,先去看看怎么使用吧,我这也简单贴一部分代码吧。

使用查询的Demo

//查询

                sql = "select * from Teacher";

                var list = SqlMapper.Query<Teacher>(conn, sql, null).ToList();



                sql = "select * from Teacher left join Student on Teacher.Id=Student.Tid";

                //一对一

                var list1 = SqlMapper.Query<Teacher, Student, Teacher>(conn, sql,

                    (t, s) =>

                    {

                        if (t.Student == null) t.Student = new List<Student>();

                        t.Student.Add(s);

                        return t;

                    }

                    , null, true, null, "Id", null, null);

                //一对多

                Dictionary<string, Teacher> list2Dict = new Dictionary<string, Teacher>();//这个才是最后的结果

                var list2 = SqlMapper.Query<Teacher, Student, Teacher>(conn, sql,

                    (t, s) =>

                    {

                        Teacher temp;

                        if (!list2Dict.TryGetValue(t.Id, out temp))

                        {

                            temp = t;

                            list2Dict.Add(temp.Id, temp);

                        }

                        if (temp.Student == null) temp.Student = new List<Student>();

                        temp.Student.Add(s);

                        return temp;

                    }

                    , null, true, null, "Id", null, null);

好了,我也不解释,自己体会。

我们先看看Dapper提供了哪些对外的查询方法呢,既然是将出参,只有查询才会涉及到DataReader转实体的呢,所以主要就看那几个查询方法就行了。

 

前两个是一个实体映射的,后面三个是多个实体映射,正常情况下一个实体对应一个表,你也可以一个表对应多个实体也是可行的。

由浅入深,先看单个实体的查询。

private static IEnumerable<T> QueryInternal<T>(IDbConnection cnn, string sql, object param, IDbTransaction transaction, int? commandTimeout, CommandType? commandType)

        {

            var identity = new Identity(sql, commandType, cnn, typeof(T), param == null ? null : param.GetType(), null);

            var info = GetCacheInfo(identity);


            using (var cmd = SetupCommand(cnn, transaction, sql, info.ParamReader, param, commandTimeout, commandType))

            {

                using (var reader = cmd.ExecuteReader())

                {

                    Func<Func<IDataReader, object>> cacheDeserializer = delegate()

                    {

                        info.Deserializer = GetDeserializer(typeof(T), reader, 0, -1, false);

                        SetQueryCache(identity, info);

                        return info.Deserializer;

                    };


                    if (info.Deserializer == null)

                    {

                        cacheDeserializer();

                    }


                    var deserializer = info.Deserializer;


                    while (reader.Read())

                    {

                        object next;

                        try

                        {

                            next = deserializer(reader);

                        }

                        catch (DataException)

                        {

                            deserializer = cacheDeserializer();

                            next = deserializer(reader);

                        }

                        yield return (T)next;

                    }

                }

            }

        }

这个就是单个实体的查询方法,用一张图片说明

很明显在读取reader的时候  next = deserializer(reader);  就是这个将reader转成实体的,那这个deserializer是什么呢,往上看啊,上面重点二字的地方就是创建deserializer 委托的地方,对了这里插一句这里委托Func(有返回值的泛型委托),之前在入参讲解的时候那里委托是Action(无返回值的泛型委托)。

也就是说deserializer就是创建委托的地方,我们去看看它的庐山真面目。

private static Func<IDataReader, object> GetDeserializer(Type type, IDataReader reader, int startBound, int length, bool returnNullIfFirstMissing)

        {

            Func<IDataReader, object> func = null;

            if (IsSimpleValue(type))

            {

                func = GetSimpleDeserializer(type, startBound);

            }

            else if (typeof(IDictionary).IsAssignableFrom(type))

            {

                func = GetDictionaryDeserializer(type, startBound);

            }

            else if (type.IsClass)

            {

                func = GetClassDeserializer(type, reader, startBound, length, returnNullIfFirstMissing);

            }

            return func;

        }

 func = GetSimpleDeserializer(type, startBound);   func = GetDictionaryDeserializer(type, startBound);  这两个是我扩展的两种类型,就是为了让出参支持简单类型和继承IDictionary的类型。

而 func = GetClassDeserializer(type, reader, startBound, length, returnNullIfFirstMissing);  这个才是重点中的难点,这个就是将reader转成实体的委托。


呵呵,我也不多说,这个就是Dapper核心价值所在,使用Emit创建实体。

上面不是说到,我也扩展出参支持两种类型嘛,使用的时候就是这样

其一:IDictionary

 sql = "select * from student";// where Id='916a84c6-85cb-4b41-b52a-96a0685d91b5'";var sex = SqlMapper.Query<Dictionary<string, object>>(conn, sql, null).ToList();

 

很简单的将结果转成 Dictionary ,它的委托实现如下:

private static Func<IDataReader, object> GetDictionaryDeserializer(Type type, int index)

        {

            return delegate(IDataReader r)

            {

                IDictionary ht = Activator.CreateInstance(type) as IDictionary;

                for (int i = 0; i < r.FieldCount; i++)

                {

                    ht.Add(r.GetName(i), r[i]);

                }

                return ht;

            };

        }

其二:SimpleValue

何为SimpleValue呢,看下面

private static bool IsSimpleValue(Type type)

        {

            if (

                        type.IsEnum ||

                        isSame(type, typeof(byte)) || isSame(type, typeof(byte?)) ||

                        isSame(type, typeof(sbyte)) || isSame(type, typeof(sbyte?)) ||

                        isSame(type, typeof(long)) || isSame(type, typeof(long?)) ||

                        isSame(type, typeof(ulong)) || isSame(type, typeof(ulong?)) ||

                        isSame(type, typeof(short)) || isSame(type, typeof(short?)) ||

                        isSame(type, typeof(ushort)) || isSame(type, typeof(ushort?)) ||

                        isSame(type, typeof(int)) || isSame(type, typeof(int?)) ||

                        isSame(type, typeof(uint)) || isSame(type, typeof(uint?)) ||

                        isSame(type, typeof(float)) || isSame(type, typeof(float?)) ||

                        isSame(type, typeof(double)) || isSame(type, typeof(double?)) ||

                        isSame(type, typeof(decimal)) || isSame(type, typeof(decimal?)) ||

                        isSame(type, typeof(char)) || isSame(type, typeof(char?)) ||

                        isSame(type, typeof(bool)) || isSame(type, typeof(bool?)) ||

                        isSame(type, typeof(DateTime)) || isSame(type, typeof(DateTime?)) ||

                        isSame(type, typeof(string)) || isSame(type, typeof(object))

            )

                return true;

            else

                return false;

        }

如果是上面的类型,我就会用下面的委托来转换reader

private static Func<IDataReader, object> GetSimpleDeserializer(Type type, int index)

        {

            return delegate(IDataReader r)

            {

                object obj = r.GetValue(index);

                if (obj == null || (obj is DBNull)) return type.IsValueType ? Activator.CreateInstance(type) : null;

                else

                {

                    if (type.IsEnum)

                        obj = Convert.ChangeType(obj, typeof(int));

                    else

                        obj = Convert.ChangeType(obj, type);

                    return obj;

                }

            };

        }

后续,上面只是讲到单个实体查询的情况,至于多个实体的查询,只需理解两个参数

public static IEnumerable<TReturn> Query<TFirst, TSecond, TReturn>(IDbConnection cnn, string sql, Func<TFirst, TSecond, TReturn> map, object param, bool buffered, IDbTransaction transaction, string splitOn, int? commandTimeout, CommandType? commandType)

 

这两个  splitOn  和  map  如果理解了这两个参数,其他跟单个实体是一样,至于这两个参数我先不讲解,后面我会提供源码,自己看,或者以后抽空我再讲讲。

 

总结:

出参就讲到这吧,后续我看讲不讲Dapper扩展的,会提供源码下载,源码可能和文章会有些出入,发布文章后我修改过源码。

相关文章: 

  • Dapper源码学习和源码修改

  • .Net开源微型ORM框架测评

  • .NET Core开发:项目实践

  • .NET Core 使用Dapper 操作MySQL

  • Dapper、Entity Framework 和混合应用

原文地址:http://www.cnblogs.com/deeround/p/6633611.html


.NET社区新闻,深度好文,微信中搜索dotNET跨平台或扫描二维码关注



本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/326252.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2020蓝桥杯省赛---java---B---6(成绩分析)

题目描述 时间限制: 1.0s 内存限制: 512.0MB 本题总分&#xff1a;15 分【问题描述】 小蓝给学生们组织了一场考试&#xff0c;卷面总分为 100 分&#xff0c;每个学生的得分都是 一个 0 到 100 的整数。请计算这次考试的最高分、最低分和平均分。【输入格式】 输入的第一行包…

2020蓝桥杯省赛---java---B---5(排序)

题目描述 思路分析 01231391&#xff0c;而01231314105。 让下标为j的字符提到最前&#xff0c;正好达到100次交换&#xff0c;这样也满足了最小字典序&#xff0c;也可以验证一下。 代码实现 package TEST;public class Main {public static void main(String[] args) {St…

ASP.NET Core MVC 源码学习:MVC 启动流程详解

前言 在 上一篇 文章中&#xff0c;我们学习了 ASP.NET Core MVC 的路由模块&#xff0c;那么在本篇文章中&#xff0c;主要是对 ASP.NET Core MVC 启动流程的一个学习。 ASP.NET Core 是新一代的 ASP.NET 应用程序&#xff0c;它是跨平台的&#xff0c;并且不依赖于 IIS&…

2020蓝桥杯省赛---java---B---9(子串分值和)

题目描述 时间限制: 3.0s 内存限制: 512.0MB 本题总分&#xff1a;25 分【问题描述】 对于一个字符串 S&#xff0c;我们定义 S 的分值 f(S) 为 S 中出现的不同的字符个 数。例如 f(”aba”) 2&#xff0c;f(”abc”) 3, f(”aaa”) 1。 现在给定一个字符串 S[0…n−1]&…

GitHub 贡献第一的微软开源软件列表

作者&#xff5c;木环 编辑&#xff5c;小智 在GitHub上贡献最多的公司&#xff0c;不是Facebook&#xff0c;也不是Google&#xff0c;而是微软。InfoQ对微软数个较受社区欢迎的项目进行了整理&#xff0c;以飨读者。希望开源的精神&#xff0c;能给技术社区带来更多的实惠&am…

android 画布控件,Android canvas画图操作之切割画布实现方法(clipRect)

本文实例讲述了Android canvas画图操作之切割画布实现方法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;android切割画布的历程不算很难&#xff0c;可是理解起来也比较麻烦&#xff0c;这里写一下我的理解 但是不一定正确&#xff1a;canvas.clipRect(30,30,70,Reg…

老司机实战Windows Server Docker:5 Windows Server Dockerfile葵花宝典

前面两篇&#xff08;简单运维1、简单运维2&#xff09;介绍了一些Windows Server Docker相关的基本运维知识。今天这一篇&#xff0c;Windows Server Dockerfile葵花宝典&#xff0c;涵盖了许多典型场景的Windows Server下的Dockerfile实例&#xff0c;并且每一个都包含可直接…

2020蓝桥杯省赛---java---B---2(寻找 2020)+测试txt

题目描述 text 0020000002202020002220002022002222202022020200022200020200222022002202202020020022200202000000002200222002022220222202220000222202200200202220200222200222202200000220220020202200022002200200200222000202220202002000000202200200220022020002022…

新起点!新征程!

好久没有更新公众号了&#xff0c;其一是因为最近这段时间有点“忙”&#xff08;迫于其他原因&#xff0c;目前包括本公众号一共运营4个公众号&#xff0c;精力不充沛&#xff09;&#xff0c;其次就是犯了懒病不想动。其中博客中也没有更新比较有质量的博文了。感觉自己堕落颓…

2020蓝桥杯省赛---java---B---3(蛇形填数)

题目描述 思路分析 找规律 看对角线 1481216 代码实现 package TEST;public class Main {public static void main(String[] args) {int res 1, t 4;for(int i2; i<20; i) {res t;t 4;//1481216}System.out.println(res); //761}}答案 761

linux微信公众号报警,zabbix报警媒介,微信报警,邮件报警

微信报警首先要申请微信企业公众号&#xff0c;创建相应应用&#xff0c;然后进行配置微信企业公众号申请&#xff0c;目前可免费前往该地址进行申请注册过程很简单&#xff0c;不信你试然后进行企业公众号的基础设置服务端报警微信脚本[rootbogon alertscripts]# pwd/usr/loca…

编写高效率的C#代码

周末空闲&#xff0c;选读了一下一本很不错的C#语言使用的书&#xff0c;特此记载下便于对项目代码进行重构和优化时查看。 Standing On Shoulders of Giants&#xff0c;附上思维导图&#xff0c;其中标记的颜色越深表示在实际中的实际意义越大。 名称内容和示例提供API时尽量…

2020蓝桥杯省赛---java---B---1(门牌制作)

题目描述 代码实现 package TEST;public class Main {public static void main(String[] args) {int sum0;for (int i 0; i < 2020; i) {int tempi;while (temp>0){if(temp%102){sum;}temp/10;}}System.out.println(sum);} }答案 624

使用Mybatis-Generator自动生成Dao、Model、Mapping相关文件

转载自 使用Mybatis-Generator自动生成Dao、Model、Mapping相关文件 Mybatis属于半自动ORM&#xff0c;在使用这个框架中&#xff0c;工作量最大的就是书写Mapping的映射文件&#xff0c;由于手动书写很容易出错&#xff0c;我们可以利用Mybatis-Generator来帮我们自动生成文…

android拦截短信获取短信内容,《英雄联盟手游》先锋测试招募说明:仅安卓用户...

招募时间&#xff1a;5月10日~5月17日测试开始时间&#xff1a;预计5月下旬或6月上旬招募(体验)要求&#xff1a;1、测试期间有较长时间可投入游戏体验&#xff1b;2、能够积极反馈和表达自己的游戏体验感受&#xff1b;3、需提前完成招募问卷(最终是否获取资格需筛选后确认)。…

ASP.NET Core MVC 源码学习:详解 Action 的匹配

前言 在 上一篇 文章中&#xff0c;我们已经学习了 ASP.NET Core MVC 的启动流程&#xff0c;那么 MVC 在启动了之后&#xff0c;当请求到达过来的时候&#xff0c;它是怎么样处理的呢&#xff1f; 又是怎么样把我们的请求准确的传达到我们的 Action 上呢&#xff1f; 那么&am…

win10偶尔打不开开始菜单(按win键和点击开始菜单都没反应)

像我这种桌面上一个图标都没有的。习惯把所有的应用程序放在开始菜单里面&#xff0c;但是……最近发现点击开始菜单或者按win键的时候召唤不出来开始菜单&#xff0c;怎么都出不来&#xff0c;怎么办&#xff1f;&#xff1f;&#xff1f;难道只有重启电脑来解决吗&#xff1f…

Mybatis 的Log4j日志输出问题 - 以及有关日志的所有问题

转载自 Mybatis 的Log4j日志输出问题 - 以及有关日志的所有问题 使用Mybatis的时候&#xff0c;有些时候能输出&#xff08;主要是指sql&#xff0c;参数&#xff0c;结果&#xff09;日志。有些时候就不能。 无法输出日志的时候&#xff0c;无论怎么配置log4j&#xff0c;…

2019蓝桥杯省赛---java---C---9(等差数列)

题目描述 代码实现 package TEST;import java.util.Arrays; import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner scanner new Scanner(System.in);int nscanner.nextInt();int[] arrnew int[n];for (int i 0; i < n; i) {arr…

Win10 Bash\/WSL调试Linux环境下的.NET Core应用程序

一、简介 使用过Mac OS的程序员都知道,在Mac Book Pro上写程序是一件比较爽的事儿,作为dotneter&#xff0c;我们都比较羡慕Mac系统的环境,比如命令行,当然设备也是挺漂亮的。 在新的Win10系统中微软给我们提供了一个基于Ubuntu的Linux子系统&#xff08;Bash/WSL&#xff09…