mysql数据库时间突然是12小时制_为什么存入mysql数据库中的timestamp,晚了13或14个小时...

# 为什么存入mysql数据库中的timestamp,晚了13个小时

## 查看数据库时区

```

show variables like '%time_zone%';

select @@global.system_time_zone;

select @@global.time_zone;

可以得到默认数据库时区:

system_time_zone | CST |

time_zone | SYSTEM|

```

## CST时区:4个含义

>CST可以为如下4个不同的时区的缩写:

>1,美国中部时间:Central Standard Time (USA) UT-6:00 ,又美国从“3月11日”至“11月7日”实行夏令时,美国中部时间改为 UT-05:00

>2,澳大利亚中部时间:Central Standard Time (Australia) UT+9:30

>3,中国标准时间:China Standard Time UT+8:00

>4,古巴标准时间:Cuba Standard Time UT-4:00

>PS:即中国标准时间UT+8,和美国UT-5,中间相差13个小时

## 查看java程序运行的本地时区

```

TimeZone.getDefault();//得到"Asia/Shanghai"

```

## debug与源码分析:

>1,测试发现,客户端到java程序端的时间戳是正确的,即通过mybatis写入数据库之前时间戳是正确的

>2,**从mybatis一路跟踪:mybatis SqlTimestampTypeHandler.setNonNullParameter()->mybatis PreparedStatement.setTimestamp-》mysql-connector preparedStatement.setTimestamp()-》preparedStatement.setTimestampInternal()-》TimeUtil.changTimestamp(),通过计算本地时区和数据库时区差值,得到数据的时间戳,再转成SimpleDateFormat.format yyyy-MM-dd HH:mm:ss格式的时间戳日期字符串,写入数据库**

>3,问题:java运行的本地时区是"Asia/Shanghai",那mysql-connector得到的数据库时区是什么样的?连接数据库的时候,mysql-connector会获取数据库的时区信息,如上数据库时区查询,得到SYSTEM,CST

## mysql-connector获取数据库时区

>1,CST 的时区是一个很混乱的时区,在与 MySQL 协商会话时区时,Java 会误以为是 CST -0500,而非 CST +0800

```

private void configureTimezone() throws SQLException {

String configuredTimeZoneOnServer = (String) this.serverVariables

.get("timezone");

if (configuredTimeZoneOnServer == null) {

configuredTimeZoneOnServer = (String) this.serverVariables

.get("time_zone");

if ("SYSTEM".equalsIgnoreCase(configuredTimeZoneOnServer)) {

configuredTimeZoneOnServer = (String) this.serverVariables

.get("system_time_zone");//得到CST,mysql-connector以为的CST是美国的CST-5:00

}

}

...

}

```

>2,TimeZone.getTimeZone(canonicalTimezone)得到CST,mysql-connector以为的CST是美国的CST-5:00,{"CST", "America/Chicago"}

>3,mysql-connector ZoneInfoFile class时区简写和时区对应关系

```

{{"ACT", "Australia/Darwin"},

{"AET", "Australia/Sydney"},

{"AGT", "America/Argentina/Buenos_Aires"},

{"ART", "Africa/Cairo"},

{"AST", "America/Anchorage"},

{"BET", "America/Sao_Paulo"},

{"BST", "Asia/Dhaka"},

{"CAT", "Africa/Harare"},

{"CNT", "America/St_Johns"},

{"CST", "America/Chicago"},

{"CTT", "Asia/Shanghai"},

{"EAT", "Africa/Addis_Ababa"},

{"ECT", "Europe/Paris"},

{"IET", "America/Indiana/Indianapolis"},

{"IST", "Asia/Kolkata"},

{"JST", "Asia/Tokyo"},

{"MIT", "Pacific/Apia"},

{"NET", "Asia/Yerevan"},

{"NST", "Pacific/Auckland"},

{"PLT", "Asia/Karachi"},

{"PNT", "America/Phoenix"},

{"PRT", "America/Puerto_Rico"},

{"PST", "America/Los_Angeles"},

{"SST", "Pacific/Guadalcanal"},

{"VST", "Asia/Ho_Chi_Minh"}};

```

## 如何解决

### 一,修改数据库时区

```

set global time_zone = '+8:00';//设置全局时区为东八区

set time_zone = '+8:00'; //

flush privileges;//刷新权限使设置立即生效

```

### 二,添加jdbc参数:serverTimezone=GMT%2B8

```

db?useUnicode=true&characterEncoding=UTF-8&useAffectedRows=true&useTimezone=true&serverTimezone=GMT%2B8

```

## 会有什么问题

>1,因为老数据是基于CST-5:00,得到的时间戳日期字符串(yyyy-MM-dd HH:mm:ss.SSS),写入数据库中,改了数据库时区或修改了JDBC的时区配置,会导致旧数据比以前慢13个小时

## 那旧数据怎么办

>1,创建一个mybatis TimstampTypehandler专门处理timestamp类型,将某个时间以前的时间戳加上13个小时的时间戳间隔,即可

```

@MappedJdbcTypes(JdbcType.TIMESTAMP)

@MappedTypes(Timestamp.class)

public class TimestampHandler extends SqlTimestampTypeHandler {

@Override

public void setNonNullParameter(PreparedStatement ps, int i, Timestamp parameter, JdbcType jdbcType)

throws SQLException {

ps.setTimestamp(i, parameter);

}

@Override

public Timestamp getNullableResult(ResultSet rs, String columnName)

throws SQLException {

//TimeZone tz=TimeZone.getDefault();

//TimeZone.setDefault(TimeZone.getTimeZone("Asia/Shanghai"));

Timestamp timestampTemp=rs.getTimestamp(columnName);

long lt=timestampTemp.getTime();

long timestampSplit=1590249600000L;//2020-05-24 00:00:00的毫秒时间戳

if(timestampSplit>lt){

Timestamp timestamp=new Timestamp(lt+13*60*60*1000);

return timestamp;

}else{

return timestampTemp;

}

}

@Override

public Timestamp getNullableResult(ResultSet rs, int columnIndex)

throws SQLException {

TimeZone.setDefault(TimeZone.getTimeZone("Asia/Shanghai"));

Timestamp timestampTemp=rs.getTimestamp(columnIndex);

long lt=timestampTemp.getTime();

long timestampSplit=1590249600000L;//2020-05-24 00:00:00的毫秒时间戳

if(timestampSplit>lt){

Timestamp timestamp=new Timestamp(lt+13*60*60*1000);

return timestamp;

}else{

return timestampTemp;

}

}

@Override

public Timestamp getNullableResult(CallableStatement cs, int columnIndex)

throws SQLException {

TimeZone.setDefault(TimeZone.getTimeZone("Asia/Shanghai"));

Timestamp timestampTemp=cs.getTimestamp(columnIndex);

long lt=timestampTemp.getTime();

long timestampSplit=1590249600000L;//2020-05-24 00:00:00的毫秒时间戳

if(timestampSplit>lt){

Timestamp timestamp=new Timestamp(lt+13*60*60*1000);

return timestamp;

}else{

return timestampTemp;

}

}

}

```

## 多人开发,timestamp时间戳使用规约

>1,接口参数涉及时间,都用时间戳,精确到秒或毫秒,全项目统一

>2,时间戳参数直接入库,不要在代码层再做一次SimpleDateFormat.format yyyy-MM-dd HH:mm:ss.SSS转换,这样会附加本地时区,导致时间戳失效,mysql connector在入库前对timestamp类型做了本地时区和数据库时区差值计算的

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/291497.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

BCVP开发者社区2022专属周边第一弹

BCVP TeamBCVP开发者社区是博主老张的哲学发起,鼓励每个人都可参与的一个分享社区,目前已经有12个参与者,19个开源项目。欢迎加入BCVP,获取专属周边礼品(文末有介绍)。官方博客还在筹建中,预计2…

转 php include

http://www.w3school.com.cn/php/php_includes.asp PHP include 实例 例子 1 假设我们有一个名为 "footer.php" 的标准的页脚文件&#xff0c;就像这样&#xff1a; <?php echo "<p>Copyright © 2006-" . date("Y") . " W3S…

在C++中调用DLL中的函数(2)

本文转自&#xff1a;http://blog.sina.com.cn/s/blog_53004b4901009h3b.html 应用程序使用DLL可以采用两种方式&#xff1a;一种是隐式链接&#xff0c;另一种是显式链接。在使用DLL之前首先要知道DLL中函数的结构信息。Visual C6.0在VC\bin目录下提供了一个名为Dumpbin.exe的…

C++之类模板最简单的使用

1、说明类模板 1) 声明类模板时要增加一行 template <class 类型参数名> template意思是“模板”,是声明类模板时必须写的关键字。在template后面的尖括号内的内容为模板的参数表列,关键字class表示其后面的是类型参数 2、写代码理解 3、运行结果 4、总结 上…

win7将 esc与 capslock 互换

一天手软&#xff0c;于是买了一个机械键盘。cherry g80-3494 红轴各方面都不错就是有一个问题我经常用vim&#xff0c;其中esc。及F键区离主键盘区实在是太远了。 习惯于vim模式的人都有一种懒症&#xff0c;就是手指非常的不喜欢就离开了主键盘区。 于是就寻思着怎么样解决这…

mysql 5.5主从同步_MySQL5.5+配置主从同步并结合ThinkPHP5设置分布式数据库

This browser does not support music or audio playback. Please play it in WeChat or another browser.前言&#xff1a;本文章是在同处局域网内的两台windows电脑&#xff0c;且MySQL是5.5以上版本下进行的一主多从同步配置&#xff0c;并且使用的是集成环境工具PHPStudy为…

C# 10的新特性

点击上方蓝字关注我们&#xff08;本文阅读所需15分钟&#xff09;我们很高兴地宣布 C# 10 作为 .NET 6 和 Visual Studio 2022的一部分已经发布了。在这篇文章中&#xff0c;我们将介绍 C# 10 的许多新功能&#xff0c;这些功能使您的代码更漂亮、更具表现力、更快。阅读 Visu…

多线程介绍和多线程模块-lock-互斥锁

多线程介绍和多线程模块线程的特点&#xff1a;线程的生命周期开始运行结束线程的退出&#xff1a;进程执行完成线程的退出方法python的系统推出模块函数start_new_thread(func, args) #(func,(name,i))allocate_lock()exit()[root133 managehosts]# vim thread01.py #!/usr/bi…

C++编译出现binding ‘const string {aka const std::__cxx11::basic_string<char>}’ to reference of type ‘std

编译异常如下&#xff1a; 解决办法&#xff1a; 我的函数是这样的 string &larger(const string &s1, const string &s2){return s1.size() > s2.size()? s1 : s2; }改成这样就行了 const string &larger(const string &s1, const string &s2){r…

POJ 3181 Dollar Dayz DP

f[i][j]f[i-j][j]f[i][j-1]&#xff0c;结果很大需要高精度。 //#pragma comment(linker, "/STACK:1024000000,1024000000") #include<cstdio> #include<cstring> #include<cstdlib> #include<algorithm> #include<iostream> #include…

JS获取本周、本季度、本月、上月的开端日期、停止日期

Js代码 /** * 获取本周、本季度、本月、上月的开端日期、停止日期 */ var now new Date(); //当前日期 var nowDayOfWeek now.getDay(); //今天本周的第几天 var nowDay now.getDate(); //当前日 var nowMonth now.getMonth(); //当前月 var nowYear now.getYea…

.NET再出发!20岁生日快乐

.NET 20周年纪念2022年是.NET20周年纪念&#xff0c;一个技术能经历20个年头&#xff0c;也说明了它的成功。想起和 .NET刚接触的时候&#xff0c;我还是一个大一的学生&#xff0c;现在也已经步入中年。作为一名80后开发者&#xff0c;我相信很多同龄人和我一样经历了中国甚至…

C++之invalid initialization of non-const reference of type ‘int’ from an rvalue of type ‘int’

1、看代码 2、编译结果 3、分析和解决 就拿f(a + b)来说,a+b的值会存在一个临时变量中,当把这个临时变量传给f时,由于f的声明中,参数是int&,不是常量引用,因为c++编译器的一个关于语义的限制。如果一个参数是以非const引用传入,c++编译器就有理由认为程序员会在函数…

mysql cbo优化器_查询优化器介绍 - PolarDB-X 云原生分布式数据库 - 阿里云

PolarDB-X接收到一条SQL后的执行过程大致如下&#xff1a;语法解析器(Parser)将SQL文本解析成抽象语法树(AST)。语法树被转化成基于关系代数的逻辑计划。优化器(Optimizer)对逻辑计划进行优化得到物理计划。执行器(Executor)执行该计划&#xff0c;得到查询结果并返回给用户。本…

这周,我们作前端,实现统一的过滤搜索

这周统一了过滤和搜索样式&#xff0c; 作个记录。 还自己写了两个css样式&#xff0c;长见识了。 filter.html {% load staticfiles %}<link rel"stylesheet" href"{% static css/select2.min.css%}" /> <link rel"stylesheet" href&q…

怎么样在线创建索引或者重构索引

重新创建&#xff08;create&#xff09;索引的主要原因是因为新的业务的发展的需要&#xff0c;而重组索引往往是因为索引的偏移膨胀或者是数据删除引起的稀疏状态&#xff0c;也就是有些人说的“碎片”&#xff0c;这个情况下&#xff0c;我们就可以在线重组索引&#xff08;…

unix网络编程之UNIX Domain Socket IPC (sockaddr_un )

socket API原本是为网络通讯设计的&#xff0c;但后来在socket的框架上发展出一种IPC机制&#xff0c;就是UNIX Domain Socket。虽然网络socket也可用于同一台主机的进程间通讯&#xff08;通过loopback地址127.0.0.1&#xff09;&#xff0c;但是UNIX Domain Socket用于IPC更有…

找最大重复次数的数和重复次数(C++ Pair)

Problem A: 第一集 你好&#xff0c;世界冠军 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 265 Solved: 50[Submit][Status][Web Board]Description “我宣布&#xff0c;第十届国际程序设计竞赛现在开始&#xff01;本次比赛时间为9点到14点……”伴随着大赛主席的宣布…

观察者模式VS发布-订阅模式

前言观察者模式的大名&#xff0c;想必各位看官早已有所耳闻。从我们现实生活来说&#xff0c;微信公众号订阅、医院挂号叫号等都属于它的实际应用。在程序世界中&#xff0c;它是一种用于将代码解耦的设计模式&#xff0c;如果你想掌握并理解这种设计模式&#xff0c;今天就和…

mysql insert replace_mysql 操作总结 INSERT和REPLACE

--他人总结的用于操作数据库的SQL一般分为两种&#xff0c;一种是查询语句&#xff0c;也就是我们所说的SELECT语句&#xff0c;另外一种就是更新语句&#xff0c;也叫做数据操作语句。言外之意&#xff0c;就是对数据进行修改。在标准的SQL中有3个语句&#xff0c;它们是INSER…