眼见不一定为实之MySQL中的不可见字符

目录

前言

一、问题的由来

1、需求背景

2、数据表结构

二、定位问题

1、初步的问题

2、编码是否有问题

3、依然回到字符本身

三、深入字符本身

1、回归本质

2、数据库解决之道

3、代码层解决

四、总结


前言

        在开始今天的博客内容之前,正在看博客的您先来看看以下这两条sql,如果您刚好还有MySQL的环境。不妨先猜测一下它输出的内容,然后看看是否有什么区别?第一条sql如下:

select length('G30L3B01') as l1;

        接着再来看另一条sql,sql脚本如下:

select length('G30L3B01') as l2;

        各位不妨猜测一下,上面两条sql语句的执行结果l1和l2分别是多少?是不是在你的预料之中。 这里不卖关子了,相信执行过sql的朋友一定看到了结果。没有数据库客户端的朋友来看我的执行结果。在给出结果之前,首先把基础环境介绍一下。这里的服务器用的的个人的Windows 7 专业版开发机,数据库服务用的是MySQL5.7,打开的查询服务窗口所在的数据库编码是UTF-8。

         上面这个结果有没有超出你的预想,这两个字符串“看”起来似乎是一模一样的。这里的“看”我打了引号,至于原因,后面会讲到。本文即以上述场景为例,讲解在MySQL中,为什么会有这种“看”起来一致,但实际上不一样的问题,通过现象找本质,通过一步一步的排查,找到问题的根源,最后在寻根溯源后,找解决的办法;从应用代码编程的角度和底层数据库的角度来解决上述问题。如果您现在也遇到了这种“看”起来不正常的值,不妨一起交流一下。

一、问题的由来

        首先依然要介绍一下上述问题的出现场景,以便于其它的朋友在此情此景下,有更大的印象。因此本节首先将对问题的场景进行详细的描述。这里打算从两个方面讲述,第一个方面是讲述需求背景,即在什么情况下作这个事。第二个是讲述数据背景,把相关的表设计也说明一下。

1、需求背景

        事情发生的背景是这样的,客户要求我们做一个功能,他们会提供一个数据的Excel模板,然后我们需要将这些数据批量导入到数据库中。这个需求咋一看起来,是一个非常简单的需求啊。读取Excel的依赖库,一大把。解析Excel的数据,然后批量插入到数据库中,对于MybatisPlus或者其它的ORM工具都是非常简单的事。当时我们选择的是阿里巴巴开源的读取Excel的组件。感兴趣的朋友可以自己上github搜索一下。然后也基于组件也已经将Excel没个表格都读取到了内存中,然后调用MP的批量插入方法,意外的是在进行批量插入的时候数据库报错了。下面还是将数据库的表先做一个介绍。

2、数据表结构

        为了实现将数据导入到数据库中,根据面向对象的原则,我们将Excel表格中的每个单元格都设计成了一个字段,然后对字段的数据类型进行了设计,同时包括数据长度。其中有一个项是输入数据的编码,然后用户是有编码规则的,每一个数据都有一个对应的编码规则。我们简单的看了他们的编码规则,得知其长度大致为8位,因此我们在数据库中设计成了varchar(8);这么设计其实是中规中矩的,非常合理。然而在上面的批量入库过程中就一直报对应的这个字段too long。这个异常看起来很奇怪,因为我们人工去“看”的时候,这个字符串的长度确实是8。于是陷入了沉思。

二、定位问题

        既然在开发过程当中出现了问题,那么如何解决问题呢?在解决问题之前,首先要定位问题,只有正确的定位问题,才能对症下药,问题才能迎刃而解。因此本节将重点讲解怎么定位问题。

1、初步的问题

        当看到上面问题的时候,第一感觉是懵的,刚开始都有点不敢相信自己的眼睛。因为肉眼根本“看”不出来这个字符串居然不是8位。最开始怀疑的是在字符串的前后可能存在空格,因此导致了其长度超出了8位。为此我们将数据进行去空,使用sql进行去空如下:


SELECT length('G30L3B01') AS VisibleLength,length(replace('G30L3B01', '​', '')) AS CleanLength;

        然后发现还是不对,经过替换后其长度还是11,说明还是超长了。 

2、编码是否有问题

        在进行空字符过滤之后还是没有解决问题。于是换了一个方向,想着有没有可能是字符集的问题。其实大家可以看看UTF-8的字符集规范,字符集虽然会有一定的影响,但是这里存储的都是英文和数字,其长度均是标准的1,因此不存在字符的问题。问题一下子没有了方向,不知道往哪个方面去排查。

3、依然回到字符本身

        在第一次尝试了空字符替换无果,又排除了字符编码的问题后,再一次将目光投入到字符本身。这一次的想法是,虽然空字符本文替换后,长度还是11,但是这并不说明其内容一定是空格。有没有什么其它的东西在捣乱呢。有时候找问题就是这样,反反复复。

三、深入字符本身

        上面从几个方面分析了可能存在的问题,尤其是第三点,我们从最开的字符又回到了起点。那么这次采用什么分析思路呢?本小节深入道来。

1、回归本质

        为了来看看这个数据到底是什么?我们将字符串转为十六进制的字符串,通过比较原始字符串来看看区别。在MySQL中可以使用HEX(str)实现转换。相关的转换SQL如下所示:

SELECT HEX('G30L3B01') AS hex_representation,HEX('G30L3B01') as n2;hex_representation	n2
4733304C33423031	4733304C33EFBBBF423031

        不知道眼尖的你发现了问题没有,上面这条SQL执行完之后发现,其目标字符串似乎不一样啊。后面的字符串,也就是有问题的字符串其长度真的超长了。下面仔细对比这两个字符串。

4733304C33        423031	
4733304C33 EFBBBF 423031

        为了方便展示,我把相同的部分进行对比,我们发现,前面的字符经过十六进制的转换后,一共16个字节,8位。是符合我们的预期的,而下面的十六进制字符则多了6字节,一共22字节,记11位。这也就是为什么这两个字符串不一样的原因。那么这个多出来的EFBBBF又是什么呢?由于这是转成了十六进制的表示,我们需要将其转为十进制。将十六进制转十进制,如果不想计算的朋友可以在网页中直接搜索:

         当然你也可以在MySQL中进行进制转换,转换方法如下:

-- CONVERT(CONV('EFBBBF', 16, 10), UNSIGNED) 是在mysql中实现将16进制转10进制数据
select length(char(15711167)),hex(char(15711167)),CONVERT(CONV('EFBBBF', 16, 10), UNSIGNED);

        执行之后,可以看到:

length(char(15711167))	hex(char(15711167))	CONVERT(CONV('EFBBBF', 16, 10), UNSIGNED)
3	EFBBBF	15711167

        到这里,为什么两个“看”起来一模一样的字符串,实际上不一样呢?根源就在这里,在字符串的中间位置插入了不可见字符。就是这不可见字符EFBBBF导致我们做数据插入时报too long。

2、数据库解决之道

        在明确了以上的问题所在之后,我们就可以根据实际情况来进行调整。只要将这种不可见字符替换掉即可。实现的方式也很简单,sql如下:

select length(REPLACE('G30L3B01', CHAR(15711167),''))  t;t
8

        通过replace函数就实现了不可见字符EFBBBF的去除,这样再调用Insert语句,就不会报too long的问题。

3、代码层解决

        除了在数据库层来解决问题,还可以在什么地方解决呢?在编码层解决是否可行。答案是肯定的。下面我们来详细介绍一下代码层的处理办法,以java语言为例。其实原理是一样的,都是要将不可见字符进行替换掉。

/*
* 字节数组转16进制字符串
*/
public static String bytesToHexString(byte[] bArr) {if (bArr == null) {return null;}StringBuffer sb = new StringBuffer(bArr.length);String sTmp;for (int i = 0; i < bArr.length; i++) {sTmp = Integer.toHexString(0xFF & bArr[i]);if (sTmp.length() < 2)sb.append(0);sb.append(sTmp);}return sb.toString();
}// 转化十六进制编码为字符串
public static String toStringHex(String s) {byte[] baKeyword = new byte[s.length() / 2];try {for (int i = 0; i < baKeyword.length; i++) {baKeyword[i] = (byte) (0xff & Integer.parseInt(s.substring(i * 2, i * 2 + 2), 16));}s = new String(baKeyword, "utf-8");// UTF-16le:Not} catch (Exception e1) {e1.printStackTrace();}return s;
}

        下面给出测试代码,您可以实际测试以下结果,看是否与我的预期一样:

public static void main(String[] args) {System.out.println("G8014Z03".substring(0, 2));System.out.println(bytesToHexString("G30L3B01".getBytes()).toUpperCase()); // 长System.out.println(bytesToHexString("G30L3B01".getBytes()).toUpperCase()); // 短System.out.println("G30L3B01".length()); // 长System.out.println("G30L3B01".length()); // 短System.out.println("进制转换======================================================");System.out.println(toStringHex("4733304C33423031"));System.out.println(toStringHex("4733304C33423031").length());System.out.println(toStringHex("4733304C33EFBBBF423031"));System.out.println(toStringHex("4733304C33EFBBBF423031").length());System.out.println("================================================");System.out.println(toStringHex("4733304C33EFBBBF423031".replaceAll("EFBBBF", "")));System.out.println(toStringHex("4733304C33EFBBBF423031".replaceAll("EFBBBF", "")).length());System.out.println("G30L3B01".equals("G30L3B01"));System.out.println(toStringHex("4733304C33EFBBBF423031".replaceAll("EFBBBF", "")).equals("G30L3B01"));}

        在代码中使用上述代码也可以实现目标字符的替换,将不可见字符进行替换掉。欢迎在实践中进行测试使用。

四、总结

        以上就是本文的主要内容,本文以数据库中不可见字符处理为例,讲解在MySQL中,为什么会有这种“看”起来一致,但实际上不一样的问题,通过现象找本质,通过一步一步的排查,找到问题的根源,最后在寻根溯源后,找到解决的办法;从应用代码编程的角度和底层数据库的角度来解决上述问题。

        使用建议,如果您也有这方面的问题,尤其是从Excel或者其它的模板中导入数据的,很有可能会遇到这个问题。如果碰到看起来一致,但实际内容不一样的情况,可以试试看是不是碰到了不可见字符,本文即分享了一种在MySQL中的不可见字符的解决方案,同时分享了使用JAVA语言进行上述问题的解决。本文行文仓促,难免有不足之处,欢迎各位专家朋友在评论区批评指正,不慎感激。

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

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

相关文章

Element-UI实现el-dialog弹框拖拽功能

在实际开发中&#xff0c;会发现有些系统&#xff0c;弹框是可以在浏览器的可见区域自由拖拽的&#xff0c;这极大方便用户的操作。但在查看Element-UI中弹框&#xff08;el-dialog&#xff09;组件的文档时&#xff0c;发现并未实现这一功能。不过也无须担心&#xff0c;vue中…

Day 28:2748. 美丽下标对的数目

Leetcode 2748. 美丽下标对的数目 给你一个下标从 0 开始的整数数组 nums 。如果下标对 i、j 满足 0 ≤ i < j < nums.length &#xff0c;如果 nums[i] 的 第一个数字 和 nums[j] 的 最后一个数字 互质 &#xff0c;则认为 nums[i] 和 nums[j] 是一组 美丽下标对 。 返回…

Linux系统之ARP命令的基本使用

Linux系统之ARP命令的基本使用 一、ARP介绍二、ARP命令帮助2.1 ARP的help帮助信息2.2 ARP命令的帮助解释 三、ARP命令的基本使用3.1 查看ARP缓存3.2 显示详细信息3.3 添加静态arp映射3.4 删除指定主机的ARP条目3.5 从文件读取并添加条目3.6 清除ARP缓存 四、注意事项五、总结 一…

wins系统资源监视器任务管理器运行监控CPU、内存、磁盘、网络运行状态

目录 1.Windows系统资源监视器的详细介绍2.通过任务管理器打开资源监视器3.任务管理中总体观察观察cpu、pid、应用程序、I/O次数或者说读写字节数 4.观察CPU观察cpu核心数&#xff0c;以及哪些占用cpu频率过高 5.观察内存观察各个应用占用的内存大小和对应线程 6.观察磁盘活动观…

【前端技巧】css篇

利用counter实现计数器 counter-reset&#xff1a;为计数器设置名称&#xff0c;语法如下&#xff1a; counter-rese: <idntifier><integer>第一个参数为变量名称&#xff0c;第二个参数为初始值&#xff0c;默认为0 counter-increment&#xff1a;设置计数器增…

LabVIEW与3D相机开发高精度表面检测系统

使用LabVIEW与3D相机开发一个高精度表面检测系统。该系统能够实时获取三维图像&#xff0c;进行精细的表面分析&#xff0c;广泛应用于工业质量控制、自动化检测和科学研究等领域。通过真实案例&#xff0c;展示开发过程中的关键步骤、挑战及解决方案&#xff0c;确保系统的高性…

宕机了, redis如何保证数据不丢?

前言 如果有人问你&#xff1a;"你会把 Redis 用在什么业务场景下&#xff1f;" 我想你大概率会说&#xff1a;"我会把它当作缓存使用&#xff0c;因为它把后端数据库中的数据存储在内存中&#xff0c;然后直接从内存中读取数据&#xff0c;响应速度会非常快。…

【Linux从入门到放弃】进程地址空间

&#x1f9d1;‍&#x1f4bb;作者&#xff1a; 情话0.0 &#x1f4dd;专栏&#xff1a;《Linux从入门到放弃》 &#x1f466;个人简介&#xff1a;一名双非编程菜鸟&#xff0c;在这里分享自己的编程学习笔记&#xff0c;欢迎大家的指正与点赞&#xff0c;谢谢&#xff01; 进…

如何更换OpenHarmony SDK API 10

OpenHarmony社区已经发布OpenHarmony SDK API 10 beta版本&#xff0c;有些 Sample案例 也有需要API10。那么如何替换使用新的OpenHarmony SDK API 10呢&#xff1f;本文做个记录。 1、如何获取OpenHarmony SDK 1.1 每日构建流水线 可以从OpenHarmony每日构建站点获取最新的…

【网络安全的神秘世界】已解决Failed to start proxy service on 127.0.0.1:8080

&#x1f31d;博客主页&#xff1a;泥菩萨 &#x1f496;专栏&#xff1a;Linux探索之旅 | 网络安全的神秘世界 | 专接本 | 每天学会一个渗透测试工具 解决burpsuite无法在 127.0.0.1&#xff1a;8080 上启动代理服务端口被占用以及抓不到本地包的问题 Burpsuite无法启动proxy…

定个小目标之刷LeetCode热题(25)

这道题采用的解法是桶排序&#xff0c;画草图如下 代码如下 //基于桶排序求解「前 K 个高频元素」 class Solution {public int[] topKFrequent(int[] nums, int k) {HashMap<Integer, Integer> map new HashMap();for (int num : nums) {if (map.containsKey(num)) {m…

【安防天下】模拟视频监控系统——模拟监控系统的构成视频采集设备

文章目录 1 模拟监控系统的构成2 视频采集设备2.1 摄像机相关技术2.1.1 摄像机的工作原理2.1.2 摄像机的分类2.1.3 摄像机的主要参数 2.2 镜头相关介绍2.2.1 镜头的主要分类2.2.2 镜头的主要参数 1 模拟监控系统的构成 模拟视频监控系统又称闭路电视监控系统&#xff0c; 一般…

htb_Blurry

端口扫描 80 按照教程注册安装clear ml 加载configuration的时候会报错 将json里的API&#xff0c;File Store的host都添加到/etc/hosts中 即可成功初始化 查找clear ml漏洞 发现一个cve-2024-24590 下面是一个利用脚本&#xff0c;但不能直接用 ClearML-vulnerability-…

好用的linux一键换源脚本

最近发现一个好用的linux一键换源脚本&#xff0c;记录一下 官方链接 大陆使用 bash <(curl -sSL https://linuxmirrors.cn/main.sh)# github地址 bash <(curl -sSL https://raw.githubusercontent.com/SuperManito/LinuxMirrors/main/ChangeMirrors.sh) # gitee地址 …

Linux基础命令大全(详解版)

Linux基础命令&#xff08;详解版&#xff09; 文章目录 Linux基础命令&#xff08;详解版&#xff09;1.Linux的目录结构**2.Linux路径的描述方式**3.Linux命令基础格式4.ls命令 隐藏文件、文件夹5.pwd命令6.cd命令 特殊路径符7.mkdir命令 文件操作命令8.touch命令9.cat命令10…

英伟达中国特供芯片降价背后:巨头与市场的较量

英伟达&#xff0c;这家曾经在人工智能芯片领域独领风骚的巨头&#xff0c;近期在中国市场遭遇了一些挑战。为了应对来自华为等中国本土企业的竞争&#xff0c;英伟达不得不采取降价策略&#xff0c;调整其专为中国市场打造的H20芯片价格&#xff0c;甚至低于华为的同类产品。这…

S级猫主食冻干测评出来了:希喂、K9、朗诺实测分享

对于许多宠物主人来说&#xff0c;一到挑选主食冻干就头疼。尽管主食冻干为猫咪带来的益处远超过普通猫粮&#xff0c;但其价格也相对较高。因此&#xff0c;许多宠物主人担心高价购买的主食冻干营养价值并不高。实际上&#xff0c;除了营养&#xff0c;安全性和配方也是选购时…

【K8s】专题五(5):Kubernetes 配置之热更新工具 Reloader

以下内容均来自个人笔记并重新梳理&#xff0c;如有错误欢迎指正&#xff01;如果对您有帮助&#xff0c;烦请点赞、关注、转发&#xff01;欢迎扫码关注个人公众号&#xff01; 目录 一、基本介绍 二、工作原理 三、部署方法 四、使用方法 一、基本介绍 Reloader 是一个用…

clickhouse学习笔记(四)库、表、分区相关DDL操作

目录 一、数据库操作 1、创建数据库 2、查询及选择数据库 3、删除数据库 二、数据表操作 1、创建表 2、删除表 3、基本操作 ①追加新字段 ②修改字段类型或默认值 ③修改字段注释 ④删除已有字段 ⑤移动数据表&#xff08;重命名&#xff09; ⑥清空表 三、默认值…

双层循环和循环语句

echo 打印 echo -n 表示不换行输出 echo -e 表示输出转义字符 echo \b 相当于退格键&#xff08;backspace&#xff09; echo \n 换行&#xff0c;相当于回车 echo \f 换行&#xff0c;换行后的新行的开头连着上一行的行尾 echo \t 相当于tab健 &#xff08;…