数据库sql中判断时间冲突

数据库,sql,时间,冲突 · 浏览次数 : 38

小编点评

**代码分析** **2.2.1代码** ```java configService.lambdaQuery() .and(q -> q) .lt(BaseTimeConfig1::getS, endTime) .gt(BaseTimeConfig1::getE, startTime) .or(q -> q) .lt(BaseTimeConfig1::getS1, endTime1) .gt(BaseTimeConfig1::getE1, startTime1) .list(); ``` **2.2.2代码** ```java configService.lambdaQuery() .and(q -> q) .lt(BaseTimeConfig1::getS, endTime) .gt(BaseTimeConfig1::getE, startTime) .or(q -> q) .lt(BaseTimeConfig1::getS1, endTime1) .gt(BaseTimeConfig1::getE1, startTime1) .list(); ``` **2.2.3代码** ```java configService.lambdaQuery() .and(q -> q) .lt(BaseTimeConfig1::getS, endTime) .gt(BaseTimeConfig1::getE, startTime) .or(q -> q) .lt(BaseTimeConfig1::getS1, endTime1) .gt(BaseTimeConfig1::getE1, startTime1) .list(); ``` **总结** 代码分析中涉及了对跨天时间段的处理以及对s,e,s,e的比较。代码中使用了lambda查询和配置服务的lambdaQuery方法来实现跨天时间的处理。

正文

数据库现有数据其中两列: s - 开始时间, e - 结束时间. 在新插入数据s', e'之前需要判断两个时间之间是否有重合
因为使用mybatis-plus的缘故, 结论都使用s或e在符号前面.

1. s < e

比如yyyy-MM-dd HH:mm:ss格式的数据, 多用于判断预约时间和每日排班冲突.
对于冲突的情况使用列举法有

  • s' < e' < s < e: 新时间段在已有时间左边, 不包含, 情况1
  • s' < s < e' < e: 新时间段和已有时间左边有交集, 情况2
  • s < s' < e' < e: 新时间段在已有时间内, 被包含关系, 也即在已有时间段内部, 情况3
  • s < s' < e < e': 新时间段和已有时间右边有交集, 情况4
  • s' < s < e < e': 新时间和已有时间是是包含关系, 也即新时间段在已有时间段外部, 情况5
  • s < e < s' < e': 新时间段在已有时间右边, 不包含, 情况6
    除去开始和最后的不包含, 可以得到当s < e' 并且 e > s'时候两个时间端肯定有交集, 也即冲突.
    image

2. 存在s > e

比如HH:mm:ss格式的, 多用于固定早中晚班定义 / 营业时间等周期性活动的时间冲突判断.

2.1 分类逐步分析

2.1.1 s < e

  • s' < e': 同1, 判断s < e' 并且 e > s'即可
  • s' > e': 也即跨天的时候, 分成s' - 240 - e'
    • 前半段: 同1; (同下面24肯定大于s, 逻辑上可以与第一种情况写到一起)
    • 下半段: 0 < e, 只需要判断s < e'即可

2.1.2 s > e

同理拆分成s - 24 和 0 - e.

  • s' < e': 不跨天
    • 前半段: 24肯定大于s', 只需要判断s < e'
    • 后半段: 0肯定小于e', 只需要判断e > s'
  • s' > e': 跨天, 拆分成s'-24 和 0-e' (这种必有24, 答案恒真)

2.1.3 代码

对于s > e情况下, 虽然sql可以使用start_time > end_time, 但是在mybatis-plus中写不出来
所以在保存时候添加一列is_greater代表是否当条数据跨天:

// 保存跨天时候的结束时间, 将结束时间变成24:00:00
// 传过来的结束时间赋值给nextDayEndTime, 也即finalNextDayEndTime
String finalNextDayEndTime = nextDayEndTime;
List<BaseTimeConfig> existList = configService.lambdaQuery()
	.and(p -> p
		// 按照开始小于结束判断(数据也是开始小于结束)
		.and(q -> q.eq(BaseTimeConfig::getIsGreater, 0)
			.and(q1 -> q1
				.lt(BaseTimeConfig::getStartTime, endTime)
				.gt(BaseTimeConfig::getEndTime, startTime))
			// 开始大于结束时间时候(), 计算拆分出来的第二天也即0点 - endTime(复制给了nextDayEndTime)
			// 需要((开始小于结束 和 开始大于结束拆分出第一天) or (开始大于结束拆分出第二天))满足其一即可
			// 拆分出来的第二天是0开始, 结束时间肯定肯定比0大, 所以判断开始时间比nextDayEndTime小即可
			.or(!"".equals(finalNextDayEndTime), q1 -> q1.lt(BaseTimeConfig::getStartTime, finalNextDayEndTime)))
		// 按照开始小于结束判断(数据是开始大于结束的, 判断结束时间大于传入的开始时间即可,
		// 因为数据库开始时间应该算0, 0肯定比传入的开始时间小于)
		.or(q -> q.eq(BaseTimeConfig::getIsGreater, 1)
			.and(q1 -> q1
				// 后半截
				.and(q2 -> q2.gt(BaseTimeConfig::getEndTime, startTime))
				// 前半截
				 .or(q2 -> q2.lt(BaseLightTimeConfig1::getStartTime, baseLightTimeConfig.getEndTime())))
			// 传入跨天的话, 数据库中跨天的都冲突
			.or(!"".equals(finalNextDayEndTime), q2 -> q2.ne(BaseLightTimeConfig1::getId, 0))))
	.ne(baseTimeConfig.getId() != null && baseTimeConfig.getId() > 0,
		BaseTimeConfig::getId, baseTimeConfig.getId())
	.isNull(BaseTimeConfig::getDeletedAt)
	.list();

2.1.4 总结

总体思路清晰, 逐步分析得到, 但不巧妙

2.2 预处理

2.2.1 提出假设

在保存s和e时候, 如果是跨天的拆分成s,24,0,e; 不跨天的话使用s,e,s,e, 对应字段s,e,s1,e1
对于s'和e'来说也是同上, 拆分出s',e',s1',e1'
分别比较s,e和s',e's1,e1和s1',e1', 两个条件使用或者连接, 也即: ((s < e' 并且 e > s') 或者 (s1 < e1' 并且 e1 > s1'))

2.2.2 分情况论证

  • 当两个都是跨天时候
    • 因为e,s1,e',s1'分别是24,0,24,0
    • 所以式子可以写为: ((s < 24 并且 24 > s') 或者 (0 < e1' 并且 e1 > 0))
    • 进一步得出: 两边都恒真, 也即两个都跨天时候肯定有重合的.
    • 0点肯定在的, 故成立. 见2.1.2下第二条
  • 当只s,e跨天时候
    • 因为e,s1,s1',e1'分别是24,0,s',e'
    • 所以式子可以写为: ((s < e' 并且 24 > s') 或者 (0 < e' 并且 e1 > s'))
    • 进一步可以得出: 只需要判断(s < e') 或者 (e1 > s')即可
    • 数据库跨天, 传入不跨, 见2.12下的第一条(e1始终是2.1中的e)
  • 当只s',e'跨天时候
    • 因为s1,e1,e',s1'分别是s,e,24,0
    • 所以式子可以写为: ((s < 24 并且 e > s') 或者 (s < e1' 并且 e > 0))
    • 进一步可以得出: 只需要判断(s < e1') 或者 (e > s')即可
    • 数据库不跨, 传入跨, 见2.1.1的第二条(e1'始终是2.1中的e')
  • 当都不跨天时候
    • 因为s1,e1,s1',e1'分别是s,e,s',e'
    • 所以式子可以写为: ((s < e' 并且 e > s') 或者 (s < e' 并且 e > s'))
    • 进一步可以得出: 只需要判断s < e' 并且 e > s'即可
    • 都不跨天, 也即都增, 见1 或 2.1.1的第一条
  • 综上, 在保存se时候, 如果保存成四个字段, 如果跨天: s,24,0,e; 如果不跨天就s,e,s,e; 同理s',e'也相同规则拆分开
    使用前面两个和前面两个比较, 后面两个与后面两个比较的结论成立.
    (一个跨天, 一个不跨天时候, 使用不跨天的必须要与跨天的两端比较, 那就不跨天时候存两遍, 这样就都比较了)

2.2.3 代码

// s', e'
String startTime = "22:00:00", endTime = "24:00:00";
// s1', e1'
String startTime1 = "00:00:00", endTime1 = "03:00:00";

// ((s < e' 并且 e > s') || (s1 < e1' 并且 e1 > s1'))
configService.lambdaQuery()
	.and(q -> q
		.lt(BaseTimeConfig1::getS, endTime)
		.gt(BaseTimeConfig1::getE, startTime))
	.or(q -> q
		.lt(BaseTimeConfig1::getS1, endTime1)
		.gt(BaseTimeConfig1::getE1, startTime1))
	.list();

2.2.4 总结

分两段存, 跨天时候24截取. 不跨天时候存两遍

与数据库sql中判断时间冲突相似的内容:

数据库sql中判断时间冲突

数据库现有数据其中两列: s - 开始时间, e - 结束时间. 在新插入数据s', e'之前需要判断两个时间之间是否有重合 因为使用mybatis-plus的缘故, 结论都使用s或e在符号前面. 1. s < e 比如yyyy-MM-dd HH:mm:ss格式的数据, 多用于判断预约时间和每日排班

SQL 注入漏洞详解 - Union 注入

1)漏洞简介 SQL 注入简介 SQL 注入 即是指 Web 应用程序对用户输入数据的合法性没有判断或过滤不严,攻击者可以在 Web 应用程序中事先定义好的查询语句的结尾上添加额外的 SQL 语句,在管理员不知情的情况下实现非法操作,以此来实现欺骗数据库服务器执行非授权的任意查询,从而进一步得到相应

小知识:IN和EXISTS的用法及效率验证

环境: Oracle 19.16 多租户架构 经常会在网上看到有人写exists和in的效率区别,其实在新版本的数据库中,是不存在这个问题的,优化器会自己判断选择最优的执行计划。 为了直观的说明,我在PDB中构造如下测试用例: vi 1.sql select count(*) from v$acti

详解Web应用安全系列(1)注入漏洞之SQL注入

注入漏洞通常是指在可输入参数的地方,通过构造恶意代码,进而威胁应用安全和数据库安全。常见的注入漏洞包括:SQL注入和XSS跨站脚本攻击。 这篇文章我们主要讲SQL注入,SQL注入即是指web应用程序对用户输入数据的合法性没有判断或过滤不严,攻击者可以在web应用程序中事先定义好的查询语句的结尾上添加

以小博大外小内大,Db数据库SQL优化之小数据驱动大数据

SQL优化中,有一条放之四海而皆准的既定方针,那就是:永远以小数据驱动大数据。其本质其实就是以小的数据样本作为驱动查询能够优化查询效率,在SQL中,涉及到不同表数据的连接、转移、或者合并,这些操作必须得有个数据集作为“带头”大哥,即驱动数据,而这个驱动数据最好是数据量最小的那一个。 内大外小 在讨论

[转帖]Oracle JDBC中的语句缓存

老熊 Oracle性能优化 2013-09-13 在Oracle数据库中,SQL解析有几种: 硬解析,过多的硬解析在系统中产生shared pool latch和library cache liatch争用,消耗过多的shared pool,使得系统不具有可伸缩性。 软解析,过多的软解析仍然可能会导

【Azure 应用服务】Java ODBC代码中,启用 Managed Identity 登录 SQL Server 报错 Managed Identity authentication is not available

问题描述 在App Service中启用Identity后,使用系统自动生成 Identity。 使用如下代码连接数据库 SQL Server: SQLServerDataSource dataSource = new SQLServerDataSource(); dataSource.setSer

MySQL高级7-存储过程

一、介绍 存储过程是事先经过编译并存储在数据库中的一段sql语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的。存储过程思想上很简单,就是数据库sql语言层面的代码封装与重用。 例如:我们在实际开发中经常会遇到先查询数据,

性能分析: 快速定位SQL问题

在数据库性能调优的实践中,SQL性能分析是至关重要的一环。一个执行效率低下的SQL语句可能会导致整个系统的性能瓶颈。 为了快速定位并解决这些问题,我们需要对SQL进行性能分析。本文将介绍一些常用的方法和技术,帮助大家快速定位SQL问题。 1、找出执行时间最长的SQL 首先,我们需要找到执行时间最长的

慢 SQL 优化之索引的作用是什么?

本文针对 MySQL 数据库的 InnoDB 存储引擎,介绍其中索引的实现以及索引在慢 SQL 优化中的作用。本文主要讨论不同场景下索引生效与失效的原因。