TRUNC两次除以0.20833333333

时间:2014-09-19 10:21:49

标签: sql oracle

我来自SQL Server的世界,并被要求弄清楚一些(古老的)Oracle脚本正在发生什么。

  

CAVEAT :我无法访问Oracle服务器,我只是有人通过电子邮件发送给我的脚本,以及我们在SQL Server上复制的数据。

     

我无权访问将数据从Oracle迁移到SQL Server的任务。我几乎是在黑暗中 - 我有Oracle脚本,以及(大部分)转换后的SQL Server表。

我在谷歌的帮助下解决了大部分内容,但我不能为我的生活弄清楚这是做什么的:

decode(TRUNC((fin_time-start_time)/0.208333333333333),0,1,1,2)

我认为我已将decode部分正确地解释为:

case TRUNC((fin_time-start_time)/0.208333333333333)
when 0 then 1
when 1 then 2
end

但TRUNC功能令我感到困惑。 endtimestarttime是没有日期部分的时间(例如16:00:0018:15:00)。同样,如果没有Oracle的访问权限或任何经验,我不知道这是否意味着有一种TIME数据类型,或者它只是一个带有'默认'日期戳的DATETIME(例如1900-01-01)。在任何一种情况下,日期都不存在或不相关。

文档似乎表明,没有格式参数传递的TRUNC()只会删除时间部分。但是时间部分是该列包含的全部内容。所以大概会导致0除以0.2083 ...我要么完全走错了轨道,要么The Daily WTF值得(或者,我怀疑两者都是)。

有什么想法吗?


更新:我认为我现在正确解释为:

case floor(cast(dateadd(minute, DATEDIFF(minute, START_TIME, FIN_TIME), 0) as float) / 0.2083333333333)
when 0 then 1
when 1 then 2
end

哪个 似乎返回了预期值,但我认为仍然是WTF,因为除了我正在正确解释DECODE()行为之外,还会返回0和1以外的值。我假设对于没有默认参数的非覆盖情况,它将返回NULL(如case那样)?

1 个答案:

答案 0 :(得分:4)

在Oracle中减去两个日期会返回一个代表天数的数字。

示例:

SELECT  TO_DATE('03.12.2004 13:00:00','DD.MM.YYYY HH24:MI:SS') 
      - TO_DATE('03.12.2004 01:00:00','DD.MM.YYYY HH24:MI:SS')
from dual ;
Result: 0.5

此表达式返回:

  • 0,如果fin_timestart_time之间的差异小于5小时,
  • 1,如果差异大于或等于5小时但小于10小时,
  • null,否则。