前言

我们知道C#中的TimeSpan对应SQL Server数据库中的Time类型,但是如果因为特殊需求数据库存储的不是Time类型,而是作为字符串,那么我们如何在查询数据时对数据库所存储的字符串类型进行比较呢?

TimeSpan类型比较

首先我们来看看正常情况下属性为TimeSpan类型进行比较的情况,给出如下实体模型。

    public class TestA
{
public int Id { get; set; }
public string StrartEnd { get; set; }
public string EndTime { get; set; } public TimeSpan TimeSpanStartEnd { get; set; }
public TimeSpan TimeSpanEndTime { get; set; }
}

            using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var testA = new TestA()
{
StrartEnd = "10:00:00",
EndTime = "22:59:00",
TimeSpanStartEnd = new TimeSpan(, , ),
TimeSpanEndTime = new TimeSpan(, , )
}; ctx.TestAs.Add(testA);
ctx.SaveChanges();
};

如上正常流程添加数据到数据库中,接下来我们来查询数据利用TimeSpanStartEnd和TimeSpanEndTime,如下:

            using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var test = ctx.TestAs.Where(d => d.TimeSpanStartEnd >= DateTime.Now.TimeOfDay && DateTime.Now.TimeOfDay <= d.TimeSpanStartEnd).ToList(); };

居然不支持TimeOfDay,需要初始化值设定项,那我们接下来初始化TimeSpan看看:

            using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var time = new TimeSpan(, , ); var test = ctx.TestAs.Where(d => d.TimeSpanStartEnd >= time && time <= d.TimeSpanStartEnd).ToList(); };

到此我们可以下一结论:在EF 6.x中对TimeSpan进行查询比较需要初始化TimeSpan值才行,否则抛出异常。问题来了,有一位园友问我,如果数据库存储的类型不是Time,而是字符串那么我们该如何查询比较呢?当时心里第一想法明明可以存储Time,为何要搞个字符串类型,结果问其原因是特殊需求,好吧,那没办法,那我们接下来探讨一下如何,请往下看,我们利用属性StartEnd和EndTime来查询。

EF 6.x如何比较TimeSpan格式的字符串?

此时我们首先需要将数据库中的字符串即StartEnd和EndTime转换为TimeSpan然后进行过滤,如下:

            using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var time = new TimeSpan(, , ); var test = ctx.TestAs.Where(d => TimeSpan.Parse(d.StrartEnd) >= time && time <= TimeSpan.Parse(d.EndTime)).ToList(); };

抛出异常无法支持TimeSpan中的Parse解析,也就是说EF 6.x无法将Parse进行翻译。在EF 6.x中对日期相关操作有SqlFunctions和DbFunctions,最终经过对相关APi的尝试,我们可通过如下改造就行,不知是否有更好的办法,一个简单的过滤,其代码实在有点多,直接上代码:

            using (var ctx = new EfDbContext())
{
ctx.Database.Log = Console.WriteLine; var time = new TimeSpan(, , ); var test = ctx.TestAs.Where(d => ((
DbFunctions.CreateTime(
SqlFunctions.DatePart("hh", d.StrartEnd),
SqlFunctions.DatePart("mi", d.StrartEnd),
SqlFunctions.DatePart("ss", d.StrartEnd)))
>= DbFunctions.CreateTime(
SqlFunctions.DatePart("hh", time),
SqlFunctions.DatePart("mi", time),
SqlFunctions.DatePart("ss", time)))
&&
((
DbFunctions.CreateTime(
SqlFunctions.DatePart("hh", time),
SqlFunctions.DatePart("mi", time),
SqlFunctions.DatePart("ss", time)))
<= DbFunctions.CreateTime(
SqlFunctions.DatePart("hh", d.EndTime),
SqlFunctions.DatePart("mi", d.EndTime),
SqlFunctions.DatePart("ss", d.EndTime)))
).ToList(); };

好了,到了这里算是给出了我在EF 6.x中实现的解决方案,Jeff在看待EF Core和EF 6.x习惯将二者拿来比较,看看EF Core是否真的强大,我们知道EF Core中没有SqlFunctions和DbFunctions两个APi,是不是就无法实现了呢,我们实践便知,同样是进行上述查询。

EF Core比较TimeSpan格式的字符串

            using (var context = new EFCoreDbContext())
{ var test = context.TestAs.Where(d => d.TimeSpanStartEnd >= DateTime.Now.TimeOfDay && DateTime.Now.TimeOfDay <= d.TimeSpanStartEnd).ToList();
}

我们能够直接比较TimeSpan类型,此时我们完全不用初始化TimeSpan直接比较就行,从最终翻译出来的SQL来看以及我在github上向EF Core团队提交的ISSUE得出利用DaTime.Now.TimeOfDay会在客户端被评估即在内存中查询,但是在EF Core 2.1会被正确翻译,详情请见我提的ISSUE(https://github.com/aspnet/EntityFrameworkCore/issues/12187)。接下来我们再来比较字符串即StartEnd和EndTime,如下:

            using (var context = new EFCoreDbContext())
{
var time = new TimeSpan(, , ); var test = context.TestAs.Where(d => TimeSpan.Parse(d.StrartEnd) >= time && time <= TimeSpan.Parse(d.EndTime)).ToList();
}

最终依然能够正常查询没有出现任何异常,相比较EF 6.x实现方式而言,代码没有那么冗长,但是依然翻译错误,无论第一种方式还是第二种方式查询在翻译成SQL时都没有添加筛选条件。

总结

在EF 6.x中对于TimeSpan类型的查询必须通过TimeSpan初始化值,否则抛出异常,如果是字符串类型的TimeSpan那么在查询时需要借助SqlFunctions和DbFunctions来实现,比较复杂,而对EF Core而言就和我们正常查询一样,没有任何异同,EF Core当前已经很稳定,还是推荐大家早早使用EF Core和.NET Core,感谢您的阅读。

特殊需求:EF 6.x如何比较TimeSpan格式的字符串?EF Core实现方式是否和EF 6.x等同?的更多相关文章

  1. 18.翻译系列:EF 6 Code-First 中的Seed Data(种子数据或原始测试数据)【EF 6 Code-First系列】

    原文链接:https://www.entityframeworktutorial.net/code-first/seed-database-in-code-first.aspx EF 6 Code-F ...

  2. EF Core 一、重识 EF

    重识EF EF Core 学习资料:https://docs.microsoft.com/zh-cn/ef/core/dbcontext-configuration/ 本为作为EF Core学习的开始 ...

  3. 【Docker】 .Net Core 3.1 webapi 集成EF Code First,使用MySql进行业务操作 、配置swagger (三)

    系列目录: [Docker] CentOS7 安装 Docker 及其使用方法 ( 一 ) [Docker] 使用Docker 在阿里云 Centos7 部署 MySQL 和 Redis (二) [D ...

  4. 项目开发中的一些注意事项以及技巧总结 基于Repository模式设计项目架构—你可以参考的项目架构设计 Asp.Net Core中使用RSA加密 EF Core中的多对多映射如何实现? asp.net core下的如何给网站做安全设置 获取服务端https证书 Js异常捕获

    项目开发中的一些注意事项以及技巧总结   1.jquery采用ajax向后端请求时,MVC框架并不能返回View的数据,也就是一般我们使用View().PartialView()等,只能返回json以 ...

  5. EF 之 MVC 排序,查询,分页 Sorting, Filtering, and Paging For MVC About EF

    最近悟出来一个道理,在这儿分享给大家:学历代表你的过去,能力代表你的现在,学习代表你的将来. 十年河东十年河西,莫欺少年穷 学无止境,精益求精    上篇博客我们学习了EF CodeFirst增删改查 ...

  6. .NET CORE 学习笔记之安装EF【Microsoft.EntityFrameworkCore】扩展报错

    最近在学习.NET CORE ,刚开始就遇到问题了. 安装EF框架的试试就报错, 报错如下: 错误 程序包还原失败.正在回滚“XXX”的程序包更改. 找了好久的方案,网上也没搜到对应的问题和方案,然而 ...

  7. 十二个 ASP.NET Core 例子——1.1版本 EF MySql快速搭建

    core1.0的时候搭建过一次mysql EF. 一大推问题.最近在core1.1 又重新搭了一次.简单搭建还挺快,没出现什么幺蛾子.总结下步骤 建立项目,例如ASP.NET Core1.1 WebA ...

  8. .NET Core 多项目工程生成EF迁移代码

    错误表现 dotnet ef的官方文档针对的是单个项目的情况,即启动项目就是DbContext所在项目. 对于分层结构的解决方案如启动项目是WebApi项目,DbContext在基础设施项目,在Web ...

  9. 【Core】在mvc使用EF

    引用DLL: 继续上一篇的内容我们来添加EF实体: 首先:工具> NuGet程序包管理器>程序包管理器控制台: Install-Package Microsoft.EntityFramew ...

随机推荐

  1. android客户端向服务器发送请求中文乱码的问

    android客户端向服务器发送请求的时候,并将参数保存到数据库时遇到了中文乱码的问题: 解决方法: url = "http://xxxx.com/Orders/saveorder.html ...

  2. ES 09 - 定制Elasticsearch的分词器 (自定义分词策略)

    目录 1 索引的分析 1.1 分析器的组成 1.2 倒排索引的核心原理-normalization 2 ES的默认分词器 3 修改分词器 4 定制分词器 4.1 向索引中添加自定义的分词器 4.2 测 ...

  3. KnockoutJS-快速入门

    虽然在WPF中接触过MVVM模式,可是刚开始在Web中接触到Knockout.JS让我大吃一惊,简化了好多工作量,原来可能需要一大堆的JS脚本完成的工作量,被释放许多.接触KnockoutJS一年多了 ...

  4. Python爬虫实例:爬取B站《工作细胞》短评——异步加载信息的爬取

    很多网页的信息都是通过异步加载的,本文就举例讨论下此类网页的抓取. <工作细胞>最近比较火,bilibili 上目前的短评已经有17000多条. 先看分析下页面 右边 li 标签中的就是短 ...

  5. jquery快速入门(五)

    jQuery - AJAX AJAX 是与服务器交换数据的技术,它在不重载全部页面的情况下,AJAX 通过后台加载数据,并在网页上进行显示,实现了对部分网页的更新. AJAX就是异步 JavaScri ...

  6. InheritableThreadLocal类原理简介使用 父子线程传递数据详解 多线程中篇(十八)

      上一篇文章中对ThreadLocal进行了详尽的介绍,另外还有一个类: InheritableThreadLocal 他是ThreadLocal的子类,那么这个类又有什么作用呢?   测试代码 p ...

  7. SQLServer事务在C#当中的应用

    1:事务是什么 事务指的是一系列SQL操作的逻辑工作单元,,要么完全地执行,要么完全地不执行. 一个逻辑工作单元必须有4个属性,原子性(Atomic).一致性(Consistent).隔离型(Isol ...

  8. 阿里、百度等多家公司Java面试记录与总结

    算算自己大概面试了近十家公司,也拿到了几个Offer,现在面试告一段落,简单总结下面试经验. 我现在主要的方向是Java服务端开发,把遇到的问题和大家分享一下,也谈谈关于技术人员如何有方向的提高自己, ...

  9. spring cloud feign 文件上传和文件下载

    文件上传参考文档:http://blog.didispace.com/spring-cloud-starter-dalston-2-4/ 文件下载参考文档:https://blog.csdn.net/ ...

  10. 如何将数据库中的值经过servlet传入到jsp页面,并且用EL表达式显示出值

    方法一:通过id查询某一数据库表中具体的行,将值封装在相应的对象中,如下面的对象Notice servlet中 String noticeId=request.getParameter("n ...