datetime和timestamp--时间戳是绝对值,日期是相对值

遇到的问题:系统时间与数据库时间不一致,系统时间是8:20,存到数据库里是0:20。

第一直觉是时区不同导致的。

先看一段代码:

Java代码

  1. public static void main(String[] args) {
  2. //System.out.println(TimeZone.getDefault());
  3. SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
  4. Calendar c = Calendar.getInstance();
  5. //System.out.println(c.getTime());
  6. System.out.println(sdf.format(c.getTime()));
  7. System.out.println(c.getTimeInMillis());
  8. TimeZone.setDefault(TimeZone.getTimeZone("GMT+5:00"));
  9. sdf.setTimeZone(TimeZone.getDefault());
  10. //System.out.println(c.getTime());
  11. //System.out.println(c.getTime().getTimezoneOffset());
  12. System.out.println(sdf.format(c.getTime()));
  13. System.out.println(c.getTimeInMillis());
  14. }

输出结果:

2011-11-25 10:33:21

1322188401796

2011-11-25 07:33:21

1322188401796

这说明,时间的显示是由时区决定的,时间所表示的距标准时间毫秒数是绝对的,不会随时区不同而改变。理解这一点很重要。

那么数据库里的时间也应该有个时区概念,那到底是如何处理的呢?一直这么认为的,时间在数据库里实际存的是毫秒数。那我们在客户端看到的时间一定是经过数据库格式化以后的。

但结果不完全是这样的,下面我们要讨论数据库中的两个时间类型 datetime 和 timestamp 。

网上有文称

datetime - 存储日期和时间部分,精确到秒,没有时区信息

timestamp - 时间戳,存储日期、时间和时区信息,秒值精确到小数点后6位

注意这里的时间戳,在sqlserver中根本就不是时间,他只记录相对时间的先后,不记录具体时间。我认为应该叫做数据版本号。

首先在sqlserver中测试:

select getdate()

执行结果跟想像的一样,操作系统的时区如何修改,他都能随之变化,他肯定获得了系统的时区信息,然后对当前毫秒数格式化。

把系统时区恢复到GMT+8:00,创建一个测试表,并插入两条数据

create table TEST_TIMEZONE(

tid int,

time1 datetime ,

time2 datetime

);

insert into TEST_TIMEZONE values(1,getdate(),getdate());

insert into TEST_TIMEZONE values(2,getdate(),getdate());

select * from TEST_TIMEZONE;

这时数据库查询结果为

1     2011-11-25 10:47:23.750       2011-11-25 10:47:23.750

2     2011-11-25 10:47:27.513       2011-11-25 10:47:27.513

现在修改系统时区为GMT+5:00,再插入两条数据,并修改第一条数据

insert into TEST_TIMEZONE values(3,getdate(),getdate());

insert into TEST_TIMEZONE values(4,getdate(),getdate());

update TEST_TIMEZONE set time1=getdate() where tid=1;

select * from TEST_TIMEZONE;

这时数据库查询结果为

1     2011-11-25 07:50:20.373       2011-11-25 10:47:23.750

2     2011-11-25 10:47:27.513       2011-11-25 10:47:27.513

3     2011-11-25 07:50:15.920       2011-11-25 07:50:15.920

4     2011-11-25 07:50:18.500       2011-11-25 07:50:18.500

因为datetime没有时区信息,只有年月日时分秒,所以保存的是几点就是几点,两次操作差了3个小时。

select t.*,t.time2-t.time1 from TEST_TIMEZONE t where t.tid=1

我们再看时间戳类型,sqlserver一个表只能有一个时间戳列,而且时间戳列不用操作,在数据行插入或更新时自动更新。

新建测试表

create table TEST_TIMEZONE2(

tid int,

time1 timestamp

);

insert into TEST_TIMEZONE2(tid) values(1);

insert into TEST_TIMEZONE2(tid) values(2);

insert into TEST_TIMEZONE2(tid) values(3);

select * from TEST_TIMEZONE2;

查询结果:

1     0x000000000000200A

2     0x000000000000200B

3     0x000000000000200C

update TEST_TIMEZONE2 set tid=4 where tid=3;

select * from TEST_TIMEZONE2;

查询结果:

1     0x000000000000200A

2     0x000000000000200B

4     0x000000000000200E

这个时间戳主要用在处理并发问题上,做为数据是否已被修改的凭证,可以提高并发性能。 再次明确sqlserver的时间戳不是具体时间。

恢复一下时区到GMT+8:00,都不知道现在几点了。

接下来,在mysql做个测试:

select now()

修改系统时区对查询结果没有影响,这与sqlserver不同。修改时区后,重启mysql,再执行有效果了。说明mysql在启动时记录了系统时区,而不是实时的读取系统时区。

恢复时区到GMT+8:00,新建 表

create table TEST_TIMEZONE(

tid int,

time1 datetime ,

time2 timestamp ,

time3 timestamp

);

插入数据

insert into TEST_TIMEZONE(tid) values(1);

insert into TEST_TIMEZONE(tid) values(2);

insert into TEST_TIMEZONE values(3,now(),now(),now());

insert into TEST_TIMEZONE values(4,now(),now(),now());

select * from TEST_TIMEZONE

查询结果

Mysql允许多个timestamp列,但只有第一列会自动更新,默认值 为

CURRENT_TIMESTAMP。

恢复时区到GMT+5:00,重启动mysql,执行查询

结果说明datetime的时间不随系统时区而变化,timestamp会随系统时区变化而变化,也sqlserver完全不同。Mysql在timestamp字段记录的是毫秒数,并且按初始的系统时区格式化后显示。

另外对oracle现在没有测试环境。

结论:

Datatime类型只保存年月日时分秒信息,不含时区。

Timestamp时间戳,不同数据库有不同的实现,不要用做业务列,更不能作为索引或键使用,他会自动被更新。

时间: 2024-10-16 03:46:46

datetime和timestamp--时间戳是绝对值,日期是相对值的相关文章

mysql中 date datetime time timestamp 的区别

MySQL中关于时间的数据类型:它们分别是 date.datetime.time.timestamp.year date :"yyyy-mm-dd"  日期     1000-01-01 到 9999-12-31   3字节 time :"hh:mm:ss" 时间     -838:59:59 到 838:59:59   3字节 datetime: "yyyy-mm-dd hh:mm:ss" 日期+时间 1000-01-01 00:00:00 到

将 javascript 中的 timestamp(时间戳) 转 c# 的 datetime类型

相信很多人在工作中都会遇到需要将 timestamp(时间戳) 转换为 datetime 类型,特别是做网站开发的时候,有时候需要将前端传递过来的 timestamp 转为 datetime 类型,而在转的过程中发现了需要注意的地方. 首先先说说在 javascript 中产生 timestamp 的方式,一般有以下几种方式: var date = new Date(); //获取timestamp var timestamp1 = date.getTime(); var timestamp2

win10 uwp unix timestamp 时间戳 转 DateTime

原文:win10 uwp unix timestamp 时间戳 转 DateTime 有时候需要把网络的 unix timestamp 转为 C# 的 DateTime ,在 UWP 可以如何转换? 转换函数可以使用下面的代码 private static DateTime UnixTimeStampToDateTime(long unixTimeStamp) { System.DateTime dtDateTime = new System.DateTime(1970, 1, 1, 0, 0,

MySql的日期类型datetime和timestamp

一.MySQL中如何表示当前时间? 其实,表达方式还是蛮多的,汇总如下: CURRENT_TIMESTAMP CURRENT_TIMESTAMP() NOW() LOCALTIME LOCALTIME() LOCALTIMESTAMP LOCALTIMESTAMP() 二.关于TIMESTAMP和DATETIME的比较 一个完整的日期格式如下:YYYY-MM-DD HH:MM:SS[.fraction],它可分为两部分:date部分和time部分,其中,date部分对应格式中的"YYYY-MM-

SpringBoot返回date日期格式化,解决返回为TIMESTAMP时间戳格式或8小时时间差

问题描述 在Spring Boot项目中,使用@RestController注解,返回的java对象中若含有date类型的属性,则默认输出为TIMESTAMP时间戳格式 ,如下所示: 解决方案       解决方案有多种,这里只记录本人觉得最简单的一种.一般我们的日期显示格式为:yyyy-MM-dd HH:mm:ss,所以我们可以在配置文件中进行全局配置. 在application.properties配置文件增加以下配置:: spring.jackson.date-format=yyyy-MM

Python datetime与timestamp之间的转换

在用Python处理datetime和timestamp的转换时发现在时区方面,Python的处理是个大坑. 因为Python的time是默认localtime输入来处理的,导致脚本在本地运行和在服务器运行会得到不一样的结果.一不注意就会中招. R中也会碰到时区的问题,但是R的方法提供了tz选项来指定日期的时区,简化了问题.而Python 中的time.mktime无法对时区进行指定,带来了很多麻烦.查了很久,发现有一个time.timezone的属性,可以根据运行脚本的机器的时间,来灵活处理时

datetime 模块详解 -- 基本的日期和时间类型

转自:https://www.cnblogs.com/fclbky/articles/4098204.html datetime 模块提供了各种类用于操作日期和时间,该模块侧重于高效率的格式化输出 在 Python 中,与时间处理有关的模块包括:time,datetime 以及 calendar datetime 模块定义了两个常量: datetime.MINYEAR - date 和 datetime 对象所能支持的最小年份,object.MINYEAR 的值为 1 datetime.MAXY

datetime,Timestamp和datetime64之间转换

引入工具包 import datetime import numpy as np import pandas as pd 总览 from IPython.display import Image from IPython.core.display import HTML Image(url= "https://i.stack.imgur.com/uiXQd.png") 一.datetime库 datetime标准库有四个主要对象 时间 - 仅限时间,以小时,分钟,秒和微秒为单位日期 -

制作时间戳和时间戳转标准日期时间等

制作时间戳:  /// <summary>        /// DateTime时间格式转换为Unix时间戳格式        /// </summary>        /// <param name="time"> DateTime时间格式</param>        /// <returns>Unix时间戳格式</returns>        public static long ConvertDate