技术标签: �й���ʱ�� time zone 时区 mysql
前言:最近,因为复习JDBC知识,使用了最新版本的mysql驱动程序(mysql-connector-java.jar),结果出现了之前从未遇到过的问题:The server time zone value ‘�й���ʱ��’ is unrecognized or represents more than one time zone。出于对未知的好奇,于是在网上查找出现该问题的具体原因,找了很久,浏览了许多文章,最常见的解答是:serverTimezone=UTC、serverTimezone=Asia/Shanghai或者降低mysql驱动程序的版本。但是,几乎没有看到有人解释问题发生的原因,也没有介绍time zone的概念,就只是给出来解决方法。在某篇博文中的评论中,我发现也有人问”为什么要这样设置(serverTimezone)“,有人回答mysql高版本驱动程序需要设置时区,这个答案让我感觉”听君一席话,如听一席话“,属于”废话文学“。最后,只能求助mysql官方文档,发现涉及到内容太多了,但是念念不忘,必有回响。坚持不懈了三天,总算对问题有了全面的认知。
系统环境:JDK 1.8、mysql-connector-java 8.0.23(mysql驱动程序)、mysql数据库 8.0.28
代码如下:
String url = "jdbc:mysql://localhost:3306/crashcourse?useUnicode=true&characterEncoding=utf8";
Class.forName("com.mysql.cj.jdbc.Driver");
Connection conn = DriverManager.getConnection(url, "root", "admin");
// 其他与数据库交互的代码省略
在IDEA运行后,出现错误,如下图所示:
报错信息中,包含了对解决问题方法的描述。
You must configure either the server or JDBC driver (via the ‘serverTimezone’ configuration property) to use a more specifc time zone value if you want to utilize time zone support.
翻译:如果你想运用时区支持,必须配置服务器或JDBC驱动程序(通过’ serverTimezone '配置属性)来使用更具体的时区值。
重启服务器会失效。使用具有权限的(配置全局变量,可以是root用户)用户,登陆到mysql服务器,运行以下命令:
mysql> set global time_zone='+8:00';
Query OK, 0 rows affected (0.00 sec)
在my.ini中添加配置项(Windwos系统中,my.ini文件位置:C:\ProgramData\MySQL\MySQL Server 8.0),重启服务器生效。
[mysqld]
default-time-zone = '+8:00'
通过serverTimezone配置连接的属性,即在url中添加serverTimezone=UTC参数。
另外,serverTimezone的值除了UTC外,还可以设置Asia/Shanghai、GMT%2b8(URL编码),把会话(也就是客户端与服务器的链接)的时区设置为东八区。
String url = "jdbc:mysql://localhost:3306/crashcourse?serverTimezone=Asia/Shanghai
String url = "jdbc:mysql://localhost:3306/crashcourse?serverTimezone=Asia/Shanghai&useTimezone=true
关于serverTimezone取值的格式:https://dev.mysql.com/doc/connector-j/8.0/en/connector-j-connp-props-datetime-types-processing.html
serverTimezone 只在客户端发挥作用,并不实际修改会话time_zone的值
例如,
JVM Connecter serverTimezone utc+9 server utc+8 底层
UTC+8 8 -9 utc+1
我觉得,严格来说,这不算是问题的解决方法,而是逃避问题的手段。如果你不想了解时区是什么及如何设置,像快点学习JDBC开发技术,这也是一种选择。但是,可能会存在软件版本之间的兼容问题。
由于mysql的java驱动程序,在版本6.0.0及以上,才要求必须配置时区,因此,可以使用6.0.0以下版本的mysql-connector-java.jar,例如、比较稳定的版本5.1.49
但是,当我查看一些高赞博文的评论时,发现有人和我存在一样的困惑:”时区是什么,为什么会有时区问题?“
如果你也有这样的疑惑,请继续阅读下面的讲解。说实话,解决这个问题,两分钟都不需要,但是真正弄清楚问题的原因,需要花费很大努力。难能可贵嘛,加油,奥利给!
”The server time zone value ‘�й���ʱ��’ is unrecognized“,服务器时区值’�й���ʱ��’是无法识别的,那么,这个乱码的值是什么?如果解决了乱码问题,是否就不会出现这个问题了呢?这是我遇到这个问题时,脑海里最初的想法。
介绍上述内容需要了解一些基础知识。
服务器系统变量(system variables),表示服务器运行中使用到配置选项。例如,配置服务器的端口号、字符集等等。可以在my.ini文件中配置,有些系统变量是动态的(dynamic),也可以在运行时使用SET 语句更改,这使你能够修改服务器的配置项而无需重启,即可生效。另外,全局系统变量(global system variables)对与所以会话(指的是一个mysql客户端到服务器的连接)都生效,会话系统变量(session system variables)只对当前会话有效,不影响其他会话。
和时区相关的系统变量:
如果设置为SYSTEM,(每个需要时区计算的 MySQL 函数调用)都会调用系统库来确定当前系统时区。
注意,两者的区别,服务器系统时区和我们的客户端程序(例如,JDBC程序)没有直接关系,而服务器当前时区是与我们的客户端程序有密切关系,我们使用的时区就是服务器当前时区。就好像character_set_server和character_set_system一样,system的编码用于服务器系统内部的编码,而server才和数据库的编码有联系。因此,会着重介绍服务器当前时区time_zone
设置全局系统变量语法:
全局系统变量,对所有会话有效,如果会话不覆盖系统变量,默认使用它
SET GLOBAL time_zone = timezone;
设置会话系统变量语法:
SET [SESSION | LOCAL | 省略] time_zone = timezone;
timezone值可以以多种格式给出,但都不区分大小写:
‘SYSTEM’,表示服务器时区与系统时区相同。
作为表示与 UTC 格式的偏移量的字符串 ,以or 为前缀,例如, 或。对于小于 10 的小时值,可以选择使用前导零;MySQL 在这种情况下存储和检索值时会添加前导零。MySQL 将 或转换为 . [H]H:MM±’+10:00’‘-6:00’‘+05:30’‘-00:00’‘-0:00’‘+00:00’在 MySQL 8.0.19 之前,这个值必须在 到 的范围内 ‘-12:59’,’+13:00’包括在内;从 MySQL 8.0.19 开始,允许的范围是’-13:59’to ‘+14:00’,包括在内。
作为命名时区,例如 ‘Europe/Helsinki’、 ‘US/Eastern’、‘MET’或’UTC’。
查看系统变量语法:
SELECT @@GLOBAL.variableName;
SELECT @@SESSION.variableName;
variableName表示变量的名字,如果像查询时区系统变量,可以SELECT @@SESSION.time_zone;
另外,如果不知道变量的具体名字,也可以模糊查询
show variables like ‘匹配的条件字符串’;
首先,使用root用户登录到服务器,执行代码:show variables like '%time_zone';
根据已有的知识,system_time_zone的时区值应该与我主机(windows10系统, 时区东八区,China Standard Time 中国标准时间。每个时区为了方便自己国际使用,都有一个名字。)的时区一致才对,然而这里显示的是空值。其实是有值的,mysql客户端对于乱码字符的处理方式有时候是选择不显示它。
执行代码: select @@global.system_time_zone;
发现果然是有值的,只是乱码而已
执行代码:set character_set_results = ‘utf8mb3’; 设置结果集编码后,在查看系统变量,发现值为"中国标准时间",这就与上文的介绍一致了。
至于,为什么设置编码为utf8mb3,这部分内容比较复杂,我会单独写一篇文章介绍。
前文介绍了,system_time_zone是和我们的客户端(JDBC程序)是无关的,如果JDBC没有设置会话的时区,默认使用全局的时区配置,也就是time_zone,上文也提到了,如果设置为’SYSTEM’,那么time_zone的时区是需要调用msyql的库函数获取的。我们的JDBC程序就是调用了msyql的库函数,返回值为“中国标准时间”。
至此,我们清楚了乱码的内容是“中国标准时间”。
对此,你也许会产生怀疑,我说JDBC程序就是调用了msyql的库函数,返回值为“中国标准时间”,实际底层是这样做的吗?如何能直接说明乱码就是“中国标准时间”呢。我来证明一下。
执行代码:select hex(@@global.system_time_zone);
hex()获取表中字符数据项目的在数据库表中的底层编码,可以看到返回值为:
(其实这是GBK编码的字符)。
(关于字符编码及乱码问题,可以参考这篇文章:《字符编码及乱码》,可以解决你日常遇到的所有乱码问题)
然后,我们使用解码工具,解码该字符编码
底层编码是GBK,但是mysql服务器却认为他是utf8mb3的,因此,必须声明解码类型为utf8mb3,而实际使用GBK解码。mysql提供的客户端程序可以做到这一点,上文中我们也看到了。但是我们的JDBC程序客户端,不可能做到这一点。因此,无法通过指定正确的编码来解决乱码问题。
先简单了解一下时区,全球被划分为24个时区,每差一个时区,区时相差一个小时。时区转换规则,“东加西减”。
例如、中国在东八区,日本在东九区,我现在的时间是22:30,那么日本现在是23:30。
时区表示方式:
UTC偏移量±[hh]:[mm]
命名表示方式,例如东八区,可以使用名字China Standard Time (或者缩写CST,中国标准时间)
当然,从上图我们也可以知道,不同的国家或者地区可以处于相同的时区,为了方便使用,不同国家或地区可以根据自己的需要对相同时区定义不同的名字。例如、我们中国定义东八区为China Standard Time,中国台湾定义东八区为Taipei Standard Time。
(Windows10系统,通过在命令行窗口运行命令:tzutil /l
可以查看系统支持的时区)
我们对什么是时区有了初步的认识,下面说明使用时区的好处。
假设,我们建立了一个发布国际新闻的网站,首先建立一张表保存新闻信息,content字段存储新闻的内容,instant字段存储发布新闻的时间。
CREATE TABLE news(
id INT UNSIGNED PRIMARY KEY,
content VARCHAR(255) NOT NULL,
instant TIMESTAMP NOT NULL
)
为了避免混乱,可以打开两个命令行窗口。
一位中国记者上传了一条新闻,设置了会话时区为东八区,然后上传了一条新闻
一位日本记者上传了一条新闻,设置会话时区为东九区,然后上传了一条新闻。
中国新闻媒体会这样报道新闻:
(注意,time_zone在会话期间,只需设置一次,我这里是为了体现命令所在的时区,因此每次执行命令,先设置时区)
设置时区后,当存储或者检索时间类型数据的时候会进行转换,依据时区设置。如果没有时区设置,那么,中国新闻媒体看到的日本记者发布的新闻时间是4月5日,0点5分。
请参考文章【mysql中的时区问题】
文章浏览阅读6.6k次,点赞8次,收藏90次。前不久有一个朋友想转去做汽车软件开发,问我需要学习什么内容。今天在这里做一个分享,希望可以帮到这位朋友以及以后想要从事汽车电子软件开发的朋友们。随着SDV(软件定义汽车)概念的提出, 大家都认识到了软件在汽车电子领域的重要性,各大汽车主机厂纷纷成立了软件研发中心并争夺软件开发人才。上汽集团在今年年初成立了上汽软件中心(后改名为零束),长城汽车成立了诺博、诺创、诺博,还有很多OEM也成立或者扩大了各自的软件研发中心,这里都不赘述了。总之,这对从事汽车软件开发的同僚们是很大的利好。2019年大家经历了汽车_汽车mcu开发怎么入门
文章浏览阅读9.2k次,点赞21次,收藏207次。对于勉强看懂了灵敏度分析的同学来说,本文给出一个方便移植、修改的python程序,可以很方便地进行灵敏度分析。_计算一阶sobol敏感性指数 python
文章浏览阅读732次。这里列出了使用jQuery构建的42个jQuery图像库、滑块、插件和教程:1. TN3 GalleryTN3 Gallery是一个完整的基于HTML的可定制的图像库,包含幻灯片、转换和多个相册选项。兼容所有现代桌面和移动浏览器。由jQuery提供。2. AD Gallery当我编写可编辑的选择插件时,我受到了自己的启发,所以我决定再写一个。这是一个与插件类似的图片库插件,比如Thickbox和L..._jthumb
文章浏览阅读1.2k次,点赞2次,收藏6次。flink 中自身虽然实现了大量的connectors,如下图所示,也实现了jdbc的connector,可以通过jdbc 去操作数据库,但是flink-jdbc包中对数据库的操作是以ROW来操作并且对数据库事务的控制比较死板,有时候操作关系型数据库我们会非常怀念在java web应用开发中的非常优秀的mybatis框架,那么其实flink中是可以自己集成mybatis进来的。 我们这里以flink 1.9版本为例来进行集成。如下图为flink内部自带的flink-jdbc:创建一个fli._flink集成mybatis
文章浏览阅读4.1k次,点赞2次,收藏21次。MySQL中的数据类型有很多,主要分为三类:1.数值类型;TINYINT 1byte (-128,127) (0,255) 小整数值SMALLINT 2bytes (-32768,32767) (0,65535) 大整数值MEDIUMINT 3bytes (-8388608,8388607) (0,16777215) 大整数值INT/INTEGER 4bytes (-2147483648, 2147483647) (0,4294967295) 大整数值BIGINT _sql数据类型
文章浏览阅读3.9k次。swagger是我们开发过程中非常常用的一个api 文档维护组织吗,为了前后端更好的交互,swagger早已经成为了大家的首选api 文档框架。但随着spring的发展与强大,spring也出了自己的api框架,但实用惯了swagger的用户,在切换过来后发现就不太会用了,其实springdoc本身已经集成并兼容了swagger,但对应的注解有所变化。下面我们就来看看swagger的注解在springdoc中的对应关系。springdoc的maven依赖 <dependency> _springdoc注解
文章浏览阅读4.3k次,点赞6次,收藏10次。1、问题描述:使用airtest 进行UI自动化测试,将python3.6 升级到python3.9。各种包安装完成后,运行脚本,报错:ImportError: cannot import name evalcontextfilter, Markup, escape from ‘jinja2‘无法从jinja2引入 evalcontextfilter, Markup, escape。2、解决办法查看了之前python3.6时的jinja2版本为3.0.3;升级到python3.9后,jinja2版本_from jinja2 import markup, escape
文章浏览阅读1.2k次,点赞2次,收藏3次。上一篇:ChatGPT+XMind,炸裂了!1.ChatGPT for google一个浏览器插件,可搭配现有的搜索引擎来使用。最大化搜索效率,对搜索体验的提升相当离谱:安装完插件后,在搜索引擎搜索任何问题,都能获取两份答案。左边是谷歌抓取的全网资源,右边是ChatGPT根据搜索结果总结提供的答案。每次检索问题都要疯狂翻页、恨不得把十几年的资讯挖个底儿透的筒子们快来集合,这个真行。2.VScode..._chatgpt软件
文章浏览阅读3.3k次。1.关于迭代器有关类型的获取与iterator_traits:有时候可能我们会需要知道迭代器的某些具体有关类型,如:迭代器所指对象的类型,两个迭代器之间距离的类型,迭代器解引用操作结果的类型,迭代器->操作结果的类型,以及迭代器本身的类型(单向,双向等)。因此,不同容器的迭代器都必须支持对上述迭代器有关类型的查询反馈,STL则采用内嵌类型声明的方式来实现对上述类型的返回。其中:value__iterator_traits
文章浏览阅读1.5k次。Java架构师交流群:793825326java版本:jdk1.8IDE:Idea2019Springboot:2.1.6.RELEASE1.pom:<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-s..._spring boot aspect原理
文章浏览阅读1.4k次,点赞2次,收藏16次。编程入门训练:题目均来自于牛客网上编程入门训练的门类,所有源码均通过平台测试,以此文记录,有助于个人学习恢复编程能力_编程训练
文章浏览阅读9.7k次,点赞4次,收藏8次。本地执行git pull, 报了如下错误:D:\IdeaProject-ws-2\shedudma\sh_offlinedata>git pullremote: Enumerating objects: 27, done.remote: Counting objects: 100% (27/27), done.remote: Compressing objects: 100% (12/12), done.remote: Total 14 (delta 10), reused 0 (delta_the stash entry is kept in case you need it again.