Mysql中处理1970年前的日期(unixtime为负数的情况)负数时间戳格式化

摘要:
“%Y”)=XXX中的生日是存储生日日期的唯一时间值的字段。PHP的date函数在处理负数的时间戳时没有问题。(注意:FROM_UNIXTIME不支持负数。请查找一个date_add函数。由于它可以用于多个date_ad函数,因此让我们总结一下date_add函数的单位(date_add(date,

客户扔过来一个bug,说是一个系统中对42岁以上的人的统计不正确,而41岁以下的人没有问题。眼睛瞟了一下托盘区里的日期,2012年3月26日,嗯,今年42岁的话,那么应该就是出生在1970年左右,马上就把问题锁定在了unixtime上,嗯,重点怀疑!

小贴士:UNIX时间,或称POSIX时间是UNIX或类UNIX系统使用的时间表示方式:从协调世界时1970年1月1日0时0分0秒起至现在的总秒数,不包括闰秒。——来自《维基百科》

看了一下数据库,生日是以int形式保存的,有正有负。这没啥问题嘛。

再看代码,SQL语句中有一个类似下面的条件语句:

WHERE FROM_UNIXTIME(birthday, '%Y') = XXX

其中的birthday是存放生日日期的unixtime值的字段,int类型的。XXX由PHP的date('Y', timestamp)来算出年份。在我的印象中,PHP的date函数处理负数的时间戳是没有问题的(注:PHP4在windows下不能处理负数时间戳,PHP5则没有这个问题。客户的服务器是Linux的,PHP4也没有问题)。好吧,那么我就开始怀疑FROM_UNIXTIME函数了,以前还真没怎么注意这玩意。随手写两个测试:

> SELECT FROM_UNIXTIME(1);
> 1970-01-01 08:00:01

看来FROM_UNIXTIME还与时区有关系。继续来:

> SELECT FROM_UNIXTIME(-1);
> NULL

哦耶,FROM_UNIXTIME果然不支持负数。测试版本为Mysql 5.1.x

这是一个条件语句,又不能拿出来由PHP处理。当然办法还是很多,比如把生日所在年换一个以unixtime表示的年龄段等等。但是我们只讨论如何用mysql计算负数的unixtime。

我们知道了unixtime表示的是从1970年1月1日0时0分0秒开始到现在的秒数,而为负的情况就是1970年第一天之前的秒数,那么可不可以通过时间计算来算出来呢?查手册,找到一个date_add函数,可以计算时间的加减。那我们就以1970年的第一天(unixtime=0的时候)作为基准来手动计算吧。由于Mysql的日期计算基本上都和时区有关系,我也懒得去搞了,就写一个大一点的秒数来看看结果:

> SELECT DATE_ADD(FROM_UNIXTIME(0), INTERVAL 1234567 SECOND);
> 1970-01-15 14:56:07

这是没有问题的,unixtime为负数的情况下:

> SELECT DATE_ADD(FROM_UNIXTIME(0), INTERVAL -1234567 SECOND);
> 1969-12-18 01:03:53

哦耶!成功了!那么剩下的问题就简单了,将本文开始的查询条件修改为:

WHERE DATE_FORMAT(DATE_ADD(FROM_UNIXTIME(0), INTERVAL birthday SECOND),'%Y') = XXX

就没问题了,BUG去除成功。

本例中用了到INTERVAL关键字,由于它不止可以用于date_add函数,那下面再总结一下DATE_ADD函数的单位(从官方手册里抄来的,为了方便理解,有一点小修改):

使用格式:INTERVAL expr type

举例:

DATE_ADD(date,INTERVAL
expr type)

DATE_SUB(date,INTERVAL expr type)

date +/-INTERVAL expr type等。

关键词INTERVA及 
type分类符均不区分大小写。

type值预期的expr格式
MICROSECOND整数
SECOND整数
MINUTE整数
HOUR整数
DAY整数
WEEK整数
MONTH整数
QUARTER整数(季节)
YEAR整数
SECOND_MICROSECOND'秒.毫秒'
MINUTE_MICROSECOND'分钟数.毫秒'
MINUTE_SECOND'分:秒'
HOUR_MICROSECOND'小时数.毫秒'
HOUR_SECOND'时:分:秒'
HOUR_MINUTE'时:分'
DAY_MICROSECOND'日期号数.毫秒'
DAY_SECOND'日 时:分:秒'
DAY_MINUTE'日 时:分'
DAY_HOUR'日 小时'
YEAR_MONTH'年-月'
 
 
 
转: https://blog.unlink.link/sql/mysql_before_1970_minus_unixtime.html
 
 
mysql的from_unixtime只能处理1970~2038年的时间戳,超过必须要上面的思路,并且int长度也只能表示到2038年,bigint可以表示的范围大些
 
Mysql中处理1970年前的日期(unixtime为负数的情况)负数时间戳格式化第1张
 
 
 
 

免责声明:文章转载自《Mysql中处理1970年前的日期(unixtime为负数的情况)负数时间戳格式化》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇使用phpMyAdmin批量修改Mysql数据表前缀的方法Vue 表情包输入组件的实现代码下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

spider爬虫练习,爬取顶点小说网,小说内容。

------------恢复内容开始------------ 我这里练习爬虫的网站是顶点小说网,地址如下: https://www.booktxt.net/ 我这里以爬取顶点小说网里面的凡人修仙传为例子: 首先观察界面: 第一章: 第二章: 由上面可得出规律,每点一次下一章,url后面的数字就会自动加1。 爬虫主要分为3步: 第一步:构建url列表...

把微信小程序异步API转为Promise,简化异步编程

把微信小程序异步API转化为Promise。用Promise处理异步操作有多方便,谁用谁知道。 微信官方没有给出Promise API来处理异步操作,而官方API异步的又非常多,这使得多异步编程会层层回调,代码一复杂,回调起来就想砸电脑。 于是写了一个通用工具,把微信官方的异步API转化为Promise,方便处理(多)异步操作。 你可以这样用: 准备转化后...

h5语音播放(移动端)

<!--语音导航 start--> <div style="border:0px solid red;100%;height:72px;position:relative;overflow-y: hidden;"> <img src="http://t.zoukankan.com/static/front/images/vo...

用Python将word文件转换成html(转)

用Python将word文件转换成html   序 最近公司一个客户大大购买了一堆医疗健康方面的科普文章,希望能放到我们正在开发的健康档案管理软件上。客户大大说,要智能推送!要掌握节奏!要深度学习!要让用户留恋网站无法自拔! 话说符合以上特点的我也只能联想到某榴了。 当然,万里长征的第一步是把文章导入我们的数据库。项目使用的是AWS的dynamoDB,是...

PHP(Mysql/Redis)消息队列的介绍及应用场景案例

在进行网站设计的时候,有时候会遇到给用户大量发送短信,或者订单系统有大量的日志需要记录,还有做秒杀设计的时候,服务器无法承受这种瞬间的压力,无法正常处理,咱们怎么才能保证系统正常有效的运行呢?这时候我们就要引用消息队列来实现这类的需求,这时候就需要一个中间的系统进行分流和解压。消息队列就是一个中间件,需要配合其他合理使用。  消息队列的概念、原理和场景 本...

highcharts配置的效果如下

配置如下: function init(categoryArray,seriesData,month_first_day,month_last_day,currDay){ var chart = Highcharts.chart('container', { chart: { type: 'columnrange...