[转载]MySQL UUID() 函数
目录
文/温国兵
一 引子
在 MySQL 中,可以有如下几种途径实现唯一值:
- 自增序列
- UUID() 函数
- 程序自定义
UUID 基于 16 进制,由 32 位小写的 16 进制数字组成,如下:
aaaaaaaa-bbbb-cccc-dddd-eeeeeeeeeeee
比如123e4567-e89b-12d3-a456-426655440000就是一个典型的 UUID。
MySQL 实现了 UUID,并且提供 UUID() 函数方便用户生成 UUID。在 MySQL 的 UUID() 函数中,前三组数字从时间戳中生成,第四组数字暂时保持时间戳的唯一性,第五组数字是一个 IEEE 802 节点标点值,保证空间唯一。使用 UUID() 函数,可以生成时间、空间上都独一无二的值。据说只要是使用了 UUID,都不可能看到两个重复的 UUID 值。当然,这个只是在理论情况下。
二 MySQL UUID() 函数
mysql -uroot -proot
mysql> SHOW VARIABLES LIKE '%version%';
+-------------------------+------------------------------+
| Variable_name | Value |
+-------------------------+------------------------------+
| innodb_version | 5.5.40 |
| protocol_version | 10 |
| slave_type_conversions | |
| version | 5.5.40-log |
| version_comment | MySQL Community Server (GPL) |
| version_compile_machine | x86_64 |
| version_compile_os | linux2.6 |
+-------------------------+------------------------------+
7 rows in set (0.00 sec)
mysql> SELECT UUID(), UUID(), LENGTH(UUID()), CHAR_LENGTH(UUID()) \G
*************************** 1. row ***************************
UUID(): 19a87b1a-a298-11e4-aa3c-08002735e4a4
UUID(): 19a87b26-a298-11e4-aa3c-08002735e4a4
LENGTH(UUID()): 36
CHAR_LENGTH(UUID()): 36
1 row in set (0.00 sec)
mysql> SELECT UUID(), UUID(), LENGTH(UUID()), CHAR_LENGTH(UUID()) \G
*************************** 1. row ***************************
UUID(): 450e1572-a298-11e4-aa3c-08002735e4a4
UUID(): 450e157c-a298-11e4-aa3c-08002735e4a4
LENGTH(UUID()): 36
CHAR_LENGTH(UUID()): 36
1 row in set (0.00 sec)
可以看到,同一个 SQL 语句中,多处调用 UUID() 函数得到的值不相同。也就是说每次调用 UUD 函数都会生成一个唯一的值。并且多次调用或执行得到的后两组值相同。另外,本身 UUID 是 32 位,因为 MySQL 生成的 UUID 有四个中划线,所以在 utf8 字符集里,长度为 36 位。
我们关闭 MySQL,然后启动。
/etc/init.d/mysqld stop
Shutting down MySQL. [ OK ]
/etc/init.d/mysqld start
Starting MySQL.. [ OK ]
再次调用 UUID() 函数。
mysql> SELECT UUID(), UUID(), LENGTH(UUID()), CHAR_LENGTH(UUID()) \G
*************************** 1. row ***************************
UUID(): 586546b2-a298-11e4-b0fc-08002735e4a4
UUID(): 586546c5-a298-11e4-b0fc-08002735e4a4
LENGTH(UUID()): 36
CHAR_LENGTH(UUID()): 36
1 row in set (0.00 sec)
可以看到,第四组的值与重启之前发生变化,直到下一次重启 MySQL。
我们连接到另一台服务器,再次调用 UUID() 函数。
mysql> SELECT UUID(), UUID(), LENGTH(UUID()), CHAR_LENGTH(UUID()) \G
*************************** 1. row ***************************
UUID(): 8fa81275-a298-11e4-8302-0800276f77f9
UUID(): 8fa81291-a298-11e4-8302-0800276f77f9
LENGTH(UUID()): 36
CHAR_LENGTH(UUID()): 36
1 row in set (0.00 sec)
可以看到跟之前的数据不同,包括第五组数据。因为第五组的值跟机器相关,所以,同一台机器第五组值不变,不同机器则变。
三 复制中的 UUID()
3.1 实验环境介绍
| 主机 | IP地址 | 主机名 | 备注 |
|---|---|---|---|
| rhel-01: | 10.10.2.231 | rhel-01 | master |
| rhel-02: | 10.10.2.227 | rhel-02 | slave |
操作系统版本:RHEL 6.5
所需要的软件包:mysql-5.5.40-linux2.6-x86_64.tar.gz
3.2 搭建复制环境
在此不赘述,请参考:MySQL AB 复制
3.3 基于 STATEMENT 模式
rhel-01 中做如下设置,设置为 STATEMENT 模式。
mysql> SET tx_isolation="REPEATABLE-READ";
Query OK, 0 rows affected (0.00 sec)
mysql> SET binlog_format="STATEMENT";
Query OK, 0 rows affected (0.00 sec)
rhel-02 也做如下设置:
mysql> SET tx_isolation="REPEATABLE-READ";
Query OK, 0 rows affected (0.00 sec)
mysql> SET binlog_format="STATEMENT";
Query OK, 0 rows affected (0.00 sec)
rhel-01 创建测试表,插入测试数据。在插入数据之后,还可以看到一个警告。
mysql> USE test;
Database changed
mysql> CREATE TABLE user
-> (name VARCHAR(36),
-> en_name VARCHAR(20),
-> job VARCHAR(10),
-> addr VARCHAR(20)
-> ) DEFAULT CHARSET=utf8 ENGINE=InnoDB;
Query OK, 0 rows affected (0.01 sec)
mysql> INSERT INTO user(name, en_name, job, addr) \
VALUES(UUID(), "robin", "dba", "GZ");
Query OK, 1 row affected, 1 warning (0.01 sec)
mysql> SHOW WARNINGS;
| Level | Code | Message |
+-------+------+-----------------------------------------------------+
| Note | 1592 | Unsafe statement written to the binary log using statement \
format since BINLOG_FORMAT = STATEMENT. Statement is unsafe because \
it uses a system function that may return a different value on the slave. |
+-------+------+-----------------------------------------------------+
1 row in set (0.00 sec)
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24d785a2-a29c-11e4-b0fc-08002735e4a4
en_name: robin
job: dba
addr: GZ
1 row in set (0.00 sec)
rhel-02 查看复制的数据。
mysql> USE test;
Database changed
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24cd38fe-a29c-11e4-8302-0800276f77f9
en_name: robin
job: dba
addr: GZ
1 row in set (0.00 sec)
可以看到,rhel-01 中的 UUID 值为24d785a2-a29c-11e4-b0fc-08002735e4a4,rhel-02 中的值为 24cd38fe-a29c-11e4-8302-0800276f77f9,两个值居然不相同,亦即主从不一致。那这样的复制是没有什么意义的。因为 UUID() 函数属于不确定函数,所以不支持 STATEMENT 模式。
3.4 基于 MIXED 模式
rhel-01 中做如下设置,设置为 MIXED 模式。
mysql> SET binlog_format="MIXED";
Query OK, 0 rows affected (0.00 sec)
rhel-02 中做如下设置:
mysql> SET binlog_format="MIXED";
Query OK, 0 rows affected (0.00 sec)
rhel-01 插入测试数据。
mysql> INSERT INTO user(name, en_name, job, addr) \
VALUES(UUID(), "Wentasy", "dba", "GZ");
Query OK, 1 row affected (0.06 sec)
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24d785a2-a29c-11e4-b0fc-08002735e4a4
en_name: robin
job: dba
addr: GZ
*************************** 2. row ***************************
name: 8dc2c93c-a29c-11e4-b0fc-08002735e4a4
en_name: Wentasy
job: dba
addr: GZ
2 rows in set (0.00 sec)
rhel-02 查看复制的数据。可以看到 MIXED 模式下,两台服务器的 UUID 相同,亦即主从一致。
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24cd38fe-a29c-11e4-8302-0800276f77f9
en_name: robin
job: dba
addr: GZ
*************************** 2. row ***************************
name: 8dc2c93c-a29c-11e4-b0fc-08002735e4a4
en_name: Wentasy
job: dba
addr: GZ
2 rows in set (0.00 sec)
3.5 基于 ROW 模式
rhel-01 中做如下设置,设置为 ROW 模式。
mysql> SET binlog_format="ROW";
Query OK, 0 rows affected (0.00 sec)
rhel-02 也做如下设置,bash mysql> SET binlog_format="ROW"; Query OK, 0 rows affected (0.00 sec)
rhel-01 插入测试数据。
mysql> INSERT INTO user(name, en_name, job, addr) \
VALUES(UUID(), "dbarobin", "dba", "GZ");
Query OK, 1 row affected (0.00 sec)
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24d785a2-a29c-11e4-b0fc-08002735e4a4
en_name: robin
job: dba
addr: GZ
*************************** 2. row ***************************
name: 8dc2c93c-a29c-11e4-b0fc-08002735e4a4
en_name: Wentasy
job: dba
addr: GZ
*************************** 3. row ***************************
name: d8123587-a29c-11e4-b0fc-08002735e4a4
en_name: dbarobin
job: dba
addr: GZ
3 rows in set (0.00 sec)
rhel-02 查看复制的测试数据。
mysql> SELECT * FROM user \G
*************************** 1. row ***************************
name: 24cd38fe-a29c-11e4-8302-0800276f77f9
en_name: robin
job: dba
addr: GZ
*************************** 2. row ***************************
name: 8dc2c93c-a29c-11e4-b0fc-08002735e4a4
en_name: Wentasy
job: dba
addr: GZ
*************************** 3. row ***************************
name: d8123587-a29c-11e4-b0fc-08002735e4a4
en_name: dbarobin
job: dba
addr: GZ
3 rows in set (0.00 sec)
可以看到,在 ROW 模式下,复制的数据和主服务器相同,亦即主从一致。
四 UUID_SHORT() 函数
在 MySQL 5.1 之后的版本,提供 UUID_SHORT() 函数,生成一个 64 位无符号整数。另外,需要注意的是,server_id 的范围必须为 0-255,并且不支持 STATEMENT 模式复制。
mysql> SELECT UUID_SHORT();
+-------------------+
| UUID_SHORT() |
+-------------------+
| 95914352036544514 |
+-------------------+
1 row in set (0.00 sec)
五 小结
- 同一个 SQL 语句中,多处调用 UUID() 函数得到的值不相同,多次调用或执行得到的后两组值相同。
- 同一台服务器,重启 MySQL 前后的 UUID() 第四组值发生变化,第五组值不变;
- MySQL 中,utf8 字符集下,生成的 UUID 长度为 36 位;
- 不同机器生成的 UUID 不同,包括第五组值;
- 在复制环境中,使用到 UUID() 函数,则一定要使用基于行或者基于混合模式复制方式。
六 Ref
–EOF–
版权声明:自由转载-非商用-非衍生-保持署名(创意共享3.0许可证)
[转载]MySQL UUID() 函数的更多相关文章
- MySQL UUID函数的详解(转)
MySQL UUID函数的详解 MySQL中可以有二类用于生成唯一值性质的工具:UUID()函数和自增序列,那么二者有何区别呢?我们就此对比下各自的特性及异同点: l 都可以实现生成唯一值的功能: ...
- [转载]MySQL concat函数的使用
MySQL concat函数是MySQL数据库中众多的函数之一,下文将对MySQL concat函数的语法和使用进行说明,供您参考和学习. MySQL concat函数使用方法:CONCAT(str1 ...
- mysql常用函数参考
mysql常用函数参考 对于针对字符串位置的操作,第一个位置被标记为1. ASCII(str) 返回字符串str的最左面字符的ASCII代码值.如果str是空字符串,返回0.如果str是NULL, ...
- 【数据库】MySql常用函数梳理
转载请注明原文地址:http://www.cnblogs.com/ygj0930/p/5866388.html MySql常用函数梳理: 1:数学函数 使用格式:一般用于插入.修改语句中,直接 函数( ...
- [php基础]Mysql日期函数:日期时间格式转换函数详解
在PHP网站开发中,Mysql数据库设计中日期时间字段必不可少,由于Mysql日期函数输出的日期格式与PHP日期函数之间的日期格式兼容性不够,这就需要根据网站实际情况使用Mysql或PHP日期转换函数 ...
- MySQL常用函数介绍
MySQL常用函数介绍 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.操作符介绍 1>.操作符优先级 mysql; +----------+ | +----------+ ...
- [转]MySQL中函数CONCAT及GROUP_CONCAT
一.CONCAT()函数 CONCAT()函数用于将多个字符串连接成一个字符串. 使用数据表Info作为示例,其中SELECT id,name FROM info LIMIT 1;的返回结果为 +-- ...
- MySQL日期函数、时间函数总结(MySQL 5.X)
一.获得当前日期时间函数 1.1 获得当前日期+时间(date + time)函数:now() select now(); # :: 除了 now() 函数能获得当前的日期时间外,MySQL 中还有下 ...
- mysql uuid使用
java中可以使用UUID类来生成uuid,使用mysql也可以使用UUID函数来获取uuid,如 select UUID(); 也可以对查询的结果做一些处理,比如说将"-"替换成 ...
随机推荐
- Apache HTTPserver安装后报:无法启动,由于应用程序的并行配置不对-(已解决)
原创作品.出自 "深蓝的blog" 博客.欢迎转载,转载时请务必注明出处.否则有权追究版权法律责任. 深蓝的blog:http://blog.csdn.net/huangyanlo ...
- HDU 1160 FatMouse's Speed (最长有序的上升子序列)
题意:给你一系列个w,s.要你找到最长的n使得 W[m[1]] < W[m[2]] < ... < W[m[n]] and S[m[1]] > S[m[2]] > ... ...
- C++ Primer高速入门之三:几种常见的控制语句
语句总是顺序运行的:第一条语句运行完了接着是第二条,第三条等等.这是最简单的情况,为了更好的控制语句的运行.程序设计语言提供了多种控制结构支持更为复杂的语句运行.我们就来看看C++ 提供的控制方式. ...
- 【quickhybrid】H5和原生的职责划分
前言 在JSBridge实现后,前端网页与原生的交互已经通了,接下来就要开始规划API,明确需要提供哪一些功能来供前端调用. 但是在这之前,还有一点重要工作需要做: 明确H5与Native的职责划分, ...
- spring boot部署系统--morphling简介
Morphling 简介 Morphling是一套基于Spring Boot 1.5开发的部署系统,依赖简单,一套Mysql即可运行,操作简单明了,适用于百台规模几下机器的运维操作 功能概述 系统部署 ...
- 理解Babel是如何编译JS代码的及理解抽象语法树(AST)
Babel是如何编译JS代码的及理解抽象语法树(AST) 1. Babel的作用是? 很多浏览器目前还不支持ES6的代码,但是我们可以通过Babel将ES6的代码转译成ES5代码,让所有的浏览器都 ...
- 【转】Android开发之数据库SQL
android中的应用开发很难避免不去使用数据库,这次就和大家聊聊android中的数据库操作. 一.android内的数据库的基础知识介绍 1.用了什么数据库 android中采用的数据库是SQLi ...
- JS中数组的迭代方法和归并方法
昨天总结的JavaScript中的数组Array方法 数组的迭代方法 ES5中为数组定义了5个迭代方法.每个方法都要接收两个参数:要在每一项上面运行的函数和(可选的)运行该函数的作用域对象---影响t ...
- JDK源码阅读(1)_简介+ java.io
1.简介 针对这一个版块,主要做一个java8的源码阅读笔记.会对一些在javaWeb中应用比较广泛的java包进行精读,附上注释.对于容易混淆的知识点给出相应的对比分析. 精读的源码顺序主要如下: ...
- PDO绑定含IN的SQL语句的参数注意事项
PDOStatement::bindParam(),表示将PDO::prepare()语句中的占位符用参数替换掉.注意,在绑定含有IN的SQL多参数语句时要额外小心,比如$stmt = $db-> ...