Mysql数据类型TINYINT(1)与BOOLEAN踩坑记
熟悉Mysql的同学应该都知道,Mysql查询的boolean结果将输出为0或者1.
比如:
select 1=1;
其输出结果为1。
查阅mysql官方文档仅找到如下描述:
11.10 Using Data Types from Other Database Engines
To facilitate the use of code written for SQL implementations from other vendors, MySQL maps data types as shown in the following table. These mappings make it easier to import table definitions from other database systems into MySQL.
Other Vendor Type MySQL Type BOOLTINYINTBOOLEANTINYINTCHARACTER VARYING(M)VARCHAR(M)FIXEDDECIMALFLOAT4FLOATFLOAT8DOUBLEINT1TINYINTINT2SMALLINTINT3MEDIUMINTINT4INTINT8BIGINTLONG VARBINARYMEDIUMBLOBLONG VARCHARMEDIUMTEXTLONGMEDIUMTEXTMIDDLEINTMEDIUMINTNUMERICDECIMAL
Other Vendor Type MySQL Type Data type mapping occurs at table creation time, after which the original type specifications are discarded. If you create a table with types used by other vendors and then issue a
DESCRIBEstatement, MySQL reports the table structure using the equivalent MySQL types. For example:tbl_namemysql> CREATE TABLE t (a BOOL, b FLOAT8, c LONG VARCHAR, d NUMERIC);
Query OK, 0 rows affected (0.00 sec) mysql> DESCRIBE t;
+-------+---------------+------+-----+---------+-------+
| Field | Type | Null | Key | Default | Extra |
+-------+---------------+------+-----+---------+-------+
| a | tinyint(1) | YES | | NULL | |
| b | double | YES | | NULL | |
| c | mediumtext | YES | | NULL | |
| d | decimal(10,0) | YES | | NULL | |
+-------+---------------+------+-----+---------+-------+
4 rows in set (0.01 sec)
我想说的是,今天使用一套中间件对kafka消息进行解析为mysql 语句,其中遇到如下的问题,
- 目标表有一字段设置类型为:tinyint(1)。
- 源表同步消息中接收到相同类型的数据。
- 其中中间件中有如下解析部分:
public void setStatement(PreparedStatement statement, DatabaseType databaseType, boolean timestampChangeToLong) throws SQLException {
if (this.value == null) {
statement.setNull(this.index, this.sqlType);
} else {
switch(this.sqlType) {
case -15:
case -9:
case 1:
case 12:
case 2005:
String strVal = String.valueOf(this.value);
statement.setString(this.index, strVal);
break;
case -7:
case 16:
boolean booleanVal = (Boolean)this.value; //tinyint(1) 类型的表设计字段直接进入该case,由于接收到的消息中的数据为0或者1,直接在该位置报类转换异常。
statement.setBoolean(this.index, booleanVal);
break;
case -6:
int val2 = (Integer)this.value;
statement.setInt(this.index, val2);
break;
case -5:
long longVal = (Long)this.value;
statement.setLong(this.index, longVal);
break;
case 2:
this.setStatementDataTypeNumeric(statement);
break;
case 3:
this.setStatementDataTypeDecimal(statement, databaseType, timestampChangeToLong);
break;
case 4:
int val = (Integer)this.value;
statement.setInt(this.index, val);
break;
case 5:
int val1 = (Integer)this.value;
statement.setInt(this.index, val1);
break;
case 6:
float floatVal = (Float)this.value;
statement.setFloat(this.index, floatVal);
break;
case 8:
double doubelVal = (Double)this.value;
statement.setDouble(this.index, doubelVal);
break;
case 91:
this.setStatementDataTypeDate(statement, databaseType);
break;
case 92:
Date timeVal = (Date)this.value;
Time sqlTime = new Time(timeVal.getTime());
statement.setTime(this.index, sqlTime);
break;
case 93:
this.setStatementDataTypeTimestamp(statement, timestampChangeToLong);
break;
default:
throw new ConsumeException("sqlType " + this.sqlType + " is not support");
} }
} - 怎样获取的数字类型呢,代码如下:
protected Database loadInternal(String database) {
Connection connection = null; Database var28;
try {
connection = this.dataSource.getConnection();//获取连接
DatabaseMetaData metaData = connection.getMetaData();//获取元数据
String catalog = null;
String[] tableTypes = new String[]{"TABLE"};
String databasePattern = this.databaseSchema != null ? this.databaseSchema : database;
ResultSet tablesResultSet = metaData.getTables((String)catalog, databasePattern, "%", tableTypes);
Database db = new Database();
db.setName(database); Table tablei;
while(tablesResultSet.next()) {
String tableName = tablesResultSet.getString("TABLE_NAME");
tablei = new Table(tableName);
db.addTable(tablei);
} Iterator var27 = db.getTables().iterator(); while(var27.hasNext()) {
tablei = (Table)var27.next();
ResultSet columnsResultSet = metaData.getColumns((String)catalog, databasePattern, tablei.getName(), (String)null); while(columnsResultSet.next()) {
String columnName = columnsResultSet.getString("COLUMN_NAME");
int sqlType = columnsResultSet.getInt("DATA_TYPE");//此处拿到mysql返回的字段类型
String typeName = columnsResultSet.getString("TYPE_NAME");
int size = columnsResultSet.getInt("COLUMN_SIZE");
boolean nullable = 1 == columnsResultSet.getInt("NULLABLE");
Column column = new Column();
column.setName(columnName);
column.setNullable(nullable);
column.setSqlType(sqlType);
column.setTypeName(typeName);
column.setSize(size);
tablei.addColumn(column);
}
} var28 = db;
} catch (Exception var25) {
throw new RuntimeException("load schema exception", var25);
} finally {
if (connection != null) {
try {
connection.close();
} catch (SQLException var24) {
;
}
} } return var28;
} - 也就是说,获取字段类型时,字段tinyint(1)的类型被当做boolean类型进行了返回。导致java中Integer类型无法进行强转。
解决方法:alter talbe change `xxx` `xxx` tinyint(4) ...;即可。修改tinyint数据类型长度,mysql也就不再当做boolean类型进行返回了。
总结:Mysql表结构设计时,要避免设计为tinyint(1)这种类型,以免与boolean类型数据结构进行混淆。引起不必要bug。当然也可以总java代码中进行修改,修改后的影响,还需另外评估。
Mysql数据类型TINYINT(1)与BOOLEAN踩坑记的更多相关文章
- Spark踩坑记——数据库(Hbase+Mysql)
[TOC] 前言 在使用Spark Streaming的过程中对于计算产生结果的进行持久化时,我们往往需要操作数据库,去统计或者改变一些值.最近一个实时消费者处理任务,在使用spark streami ...
- [转]Spark 踩坑记:数据库(Hbase+Mysql)
https://cloud.tencent.com/developer/article/1004820 Spark 踩坑记:数据库(Hbase+Mysql) 前言 在使用Spark Streaming ...
- Spark踩坑记——数据库(Hbase+Mysql)转
转自:http://www.cnblogs.com/xlturing/p/spark.html 前言 在使用Spark Streaming的过程中对于计算产生结果的进行持久化时,我们往往需要操作数据库 ...
- Spark踩坑记——共享变量
[TOC] 前言 Spark踩坑记--初试 Spark踩坑记--数据库(Hbase+Mysql) Spark踩坑记--Spark Streaming+kafka应用及调优 在前面总结的几篇spark踩 ...
- Spark踩坑记——从RDD看集群调度
[TOC] 前言 在Spark的使用中,性能的调优配置过程中,查阅了很多资料,之前自己总结过两篇小博文Spark踩坑记--初试和Spark踩坑记--数据库(Hbase+Mysql),第一篇概况的归纳了 ...
- Spark踩坑记——Spark Streaming+Kafka
[TOC] 前言 在WeTest舆情项目中,需要对每天千万级的游戏评论信息进行词频统计,在生产者一端,我们将数据按照每天的拉取时间存入了Kafka当中,而在消费者一端,我们利用了spark strea ...
- Spark踩坑记:Spark Streaming+kafka应用及调优
前言 在WeTest舆情项目中,需要对每天千万级的游戏评论信息进行词频统计,在生产者一端,我们将数据按照每天的拉取时间存入了Kafka当中,而在消费者一端,我们利用了spark streaming从k ...
- Vue + TypeScript + Element 搭建简洁时尚的博客网站及踩坑记
前言 本文讲解如何在 Vue 项目中使用 TypeScript 来搭建并开发项目,并在此过程中踩过的坑 . TypeScript 具有类型系统,且是 JavaScript 的超集,TypeScript ...
- WinUI 3 踩坑记:第一个窗口
本文是 WinUI 3 踩坑记 的一部分,该系列发布于 GitHub@Scighost/WinUI3Keng,文中的代码也在此仓库中,若内容出现冲突以 GitHub 上的为准. WinUI 3 应用的 ...
随机推荐
- CF 988C Equal Sums 思维 第九题 map
Equal Sums time limit per test 2 seconds memory limit per test 256 megabytes input standard input ou ...
- 常用logback.xml配置详解
选择logback的理由 ==logback==与==log4j==的简单对比一下: 1.首先,对于同样的代码路径,==logback==使用起来更快. 2.==logback==原生实现了log4j ...
- RobotFramework自动化测试框架-MongoDBLibrary库的使用
笔者接着 RobotFramework自动化测试框架-DatabaseLibrary库的使用(对数据库的操作) 继续分享robotframework 对数据库中的MongoDB的详细操作. Mongo ...
- Apache JMeter (一)环境的配置和操作
JMeter是Apache组织的开放源代码项目,是一款优秀的开源测试工具,可以做功能测试和性能测试.是每个资深的测试工程师必须掌握的测试工具,熟悉JMeter可以大大提高工作效率. 1.下载Jmete ...
- 1512: [POI2006]Pro-Professor Szu
首先把边反向, 问题转化成求从主建筑楼走向各个点的方案数. 然后缩点,块中的方案数可以直接算. 设f[i]表示走到第i个点的方案数.显然f[i]=∑f[j](存在newedge(j,i))初始时,f[ ...
- BASK、BFSK、BPSK调制方法的Matlab程序实现
%以下为手动编程方法,也可调用matlab集成函数dmod,具体调制方式见doc. n = [0:0.01:5.99]; x1 = ones(1,100); x2 = zeros(1,100); x3 ...
- 巨杉Tech | Hbase迁移至SequoiaDB 实战
背景 在传统银行 IT 架构中,联机交易与统计分析系统往往采用不同的技术与物理设备,通过定期执行的 ETL 将联机交易数据向分析系统中迁移.而作为数据服务资源池,同一份数据可能被不同类型的微服务共享访 ...
- springboot postman 对象里传时间格式问题
主要问题是系列化的问题,在定义时间变量处使用如下的注解即可.导包注意了······ 如果springmvc使用的是com.fasterxml.jackson 的jar包则直接使用一下注解即可 impo ...
- thinkphp6.0 集成Alipay 手机和电脑端支付的方法
本文由 BI8EJM 原创,转载请注明出处! 第一步 下载 Alipay 的PHP SDK :https://docs.open.alipay.com/54/103419/ 第二步 解压下载都到的压 ...
- 删除mac开机启动项
1、开“系统偏好设置”窗口,选择“用户与群组”,进入用户与群组窗口.选择登录项选项卡,再解锁,最后删除开机启动的应用. 2、分别在以下6个目录中检查是否有与anydesk相关的plist文件 ~/ ...