【已解决】MySQL:执行存储过程报错(MySQL字符集和排序方式冲突)

目录

问题现象:

问题分析:

解决方法:

拓展:

1、转换条件两边的字段或值为二进制数据:

 2、转换条件两边的字段或值的字符集和排序方式:

3、修改列、表、库的字符集和排序方式

参考链接:


问题现象:

        今天在执行Mysql的存储过程的时候,发现了一个意料之外的报错,如下:


问题分析:

        起因是因为最近项目要做系统演示,需要不断造数据和删数据,但由于部分业务功能还未完善,因此删数据的操作,还需要手动去数据库删除,所以为了方便运营人员和测试人员的操作,我就写了一个脚本。

        又因为需要删除的数据并非仅仅来源于单表,有多个表的数据都需要操作,因此就涉及到事务问题,最终决定用存储过程来解决,同时也能避免操作者在连续执行多个sql执行时,因为系统卡顿、网络、工具等各种可抗力或不可抗力导致脚本执行不彻底,而影响到系统演示,毕竟是给领导汇报,打脸的事,大家都不想啊!!!

        回到正题,文章开头提到的问题到底是怎么一回事呢?        根据报错信息可知,这是由于MySQL字符集和排序方式冲突导致的报错。

        下面是完整的存储过程:

-- 删除存储过程
DROP PROCEDURE IF EXISTS deleteStaffAppRelationship;

-- 创建一个存储过程:
DELIMITER $$
CREATE PROCEDURE deleteStaffAppRelationship(IN param_phone VARCHAR(255), OUT result INT(1))
BEGIN     
-- -- 如果出现异常,抛出一个sql状态码为'23000'的异常
	DECLARE EXIT HANDLER FOR SQLSTATE '23000' set result = -1;

	-- 初始化出参值
	set result = 0;

	delete from app_sys.asys_user_binding
	where USER_ID in (
		select id 
		from app_sys.asys_user
		where phone = param_phone
	)
	and IS_DELETED = 0;

    -- 其他的sql......


    -- 设置出参值
	set result = 1;
END; $$
DELIMITER;

         经过测试发现,报错原因是在如下sql中:

	delete from app_sys.asys_user_binding
	where USER_ID in (
		select id 
		from app_sys.asys_user
		where phone = param_phone
	)
	and IS_DELETED = 0;

        更准确的说就是在这个条件语句:

where phone = param_phone

        查询数据库字符集和排序方式:

-- 查看数据库字符集
show VARIABLES like '%character%';

-- 查看数据库排序方式
show VARIABLES where Variable_name like 'collation%';

        可以看到排序方式并不是完全一致的,在调用存储过程时,传入的参数param_phone使用的排序方式是:utf8mb4_0900_ai_ci,而表字段phone使用的排序方式是:utf8mb4_0900_ai_ci,因此在做条件判断时,就会因为排序方式不同而发生冲突,导致报错:

CALL app_sys.deleteStaffAppRelationship('123',@result)
> 1267 - Illegal mix of collations (utf8mb4_general_ci,IMPLICIT) and (utf8mb4_0900_ai_ci,IMPLICIT) for operation '='
> 时间: 0.01s

        既然知道问题原因,那就有解决思路了:

        只要保证两边的字符集和排序方式一致即可。

解决方法:

        有很多方法(在下文的拓展章节中会说明)可以保证字符集和排序方式的一致,所以建议根据实际情况来选择;这里先给出我目前使用的方法:

        转换条件两边的字段或值为二进制数据:

        修改后的脚本如下:

-- 删除存储过程
DROP PROCEDURE IF EXISTS deleteStaffAppRelationship;

-- 创建一个存储过程:
DELIMITER $$
CREATE PROCEDURE deleteStaffAppRelationship(IN param_phone VARCHAR(255), OUT result INT(1))
BEGIN     
-- -- 如果出现异常,抛出一个sql状态码为'23000'的异常
	DECLARE EXIT HANDLER FOR SQLSTATE '23000' set result = -1;

	-- 初始化出参值
	set result = 0;

	delete from app_sys.asys_user_binding
	where USER_ID in (
		select id 
		from app_sys.asys_user
		where binary phone = binary param_phone 
	)
	and IS_DELETED = 0;

    -- 其他的sql......


    -- 设置出参值
	set result = 1;
END; $$
DELIMITER;

        执行存储过程成功:


拓展:

        上文提到有很多方法可以,这里就简单列举一下:

1、转换条件两边的字段或值为二进制数据

where binary 字段或值 = binary 字段或值

-- 如:
where binary phone = binary param_phone

        转为二进制后的数据,相当于字符集和排序方式相同,可以直接比较。

        如果只是临时涉及到字符集或排序方式冲突问题时,建议使用这种方式。

 2、转换条件两边的字段或值的字符集和排序方式:

where CONVERT(字段或值 USING utf8mb4) COLLATE utf8mb4_general_ci = CONVERT(字段或值 USING utf8mb4) COLLATE utf8mb4_general_ci 

--如:
where CONVERT(phone USING utf8mb4) COLLATE utf8mb4_general_ci = CONVERT(param_phone USING utf8mb4) COLLATE utf8mb4_general_ci

        通过转换,保证字符集和排序方式的一致即可比较。

        这是最简单易懂的方式,但也是写起来最麻烦的方式,同样适用于临时涉及到字符集或排序方式冲突问题时。

3、修改列、表、库的字符集和排序方式

        可以通过数据库工具进行相关操作;也可以通过sql:

-- 列:
ALTER TABLE 表名 MODIFY 列名 列的数据类型 CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;

-- 如:
ALTER TABLE app_sys.asys_user_binding MODIFY phone VARCHAR(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;




-- 表:
ALTER TABLE 表名 CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;

-- 如:
ALTER TABLE app_sys.asys_user_binding CONVERT TO CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;




-- 库:
ALTER DATABASE 库名 CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;

-- 如:
ALTER DATABASE app_sys CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci;

        这是一劳永逸的最根本的方式,但一般不建议直接修改这些已经定好的字符集和排序方式,除非是拥有相应的权限,否则建议和团队中的技术领导讨论,另外网上还有一个说法指出:修改后只对以后插入的数据有效,对已有数据不生效(未亲测,有所以后验证)。


参考链接:

mysql 1267 - Illegal mix of collations (utf8mb4_general_ci,IMPLICIT) and_Marydon的技术博客_51CTO博客

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于Git的代码工程管理——学习记录一

一、Git简概[1] Git是一个分布式版本控制系统,它跟踪任何一组计算机文件的更改,通常用于在软件开发过程中协调协作开发源代码的程序员之间的工作。其为实现快速、数据完整性以及分布式非线性工作流程(在不同计算机上运行数千个并行分支&#…

电脑上mp4视频文件无缩略图怎么办

前言:有时候电脑重装后电脑上的mp4视频文件无缩略图,视频文件数量比较多的时候查找比较麻烦 以下方法亲测有效: 1、下载MediaPreview软件 2、软件链接地址:https://pan.baidu.com/s/1bzVJpmcHyGxXNjnzltojtQ?pwdpma0 提取码&…

解密IIS服务器API跨域问题的终极解决方案

在当今数字化时代,API已成为现代应用程序的核心组件。然而,当你使用IIS(Internet Information Services)服务器提供API时,你可能会遇到一个常见的挑战:API跨域问题。这个问题经常困扰着开发人员&#xff0c…

基于springboot的滑雪场管理系统源码

🍅 简介:500精品计算机源码学习,有8个项目关注搏主即可领取。另送简历模板、答辩模板、学习资料、答辩常见问题【关注我,都给你】 🍅 欢迎点赞 👍 收藏 ⭐留言 📝 文末获取源码 目录 一、以下学…

java学习part37定制排序和自然排序

150-常用类与基础API-使用Comparator接口实现定制排序及对比_哔哩哔哩_bilibili 1.自然排序 2.定制排序 对于一些排序方法,允许传入的话按定制的排序规则来,不传入默认按自然排序来。 匿名方式 3区别

JavaWeb-JavaScript

一、什么是JavaScript JavaScript是由网景的LiveScript发展而来的客户端脚本语言,主要目的是为了解决服务端语言遗留的速度问题,为客户提供更流畅的浏览效果。JavaScript可以实现网页内容、数据的动态变化和动画特效等。JavaScript的标准由ECMA维护&…

超大规模集成电路设计----CMOS反相器(五)

本文仅供学习,不作任何商业用途,严禁转载。绝大部分资料来自----数字集成电路——电路、系统与设计(第二版)及中国科学院段成华教授PPT 超大规模集成电路设计----CMOS反相器(五) 5.1 静态CMOS反相器综述5.1.1 静态CMOS反相器优点…

Excel——TEXTJOIN函数实现某一列值相等时合并其他列

一、TEXTJOIN函数介绍 公式TEXTJOIN(分隔符, 忽略空白单元格, 字符串1…) 分隔符:文本字符串,或者为空,或用双引号引起来的一个或多个字符,或对有效文本字符串的引用。如果提供一个数字,则将被视为文本。 忽略空白单…

【论文笔记】A Transformer-based Approach for Source Code Summarization

A Transformer-based Approach for Source Code Summarization 1. Introduction2. Approach2.1 ArchitectureSelf-AttentionCopy Attention 2.2 Position Representations编码绝对位置编码成对关系 1. Introduction 生成描述程序功能的可读摘要称为源代码摘要。在此任务中&…

卡通渲染总结《一》

本文是在看完之前的综述论文《Cartoon Style Rendering》的总结,论文时间是2008年有点早,但有一定启发意义。 前言 首先卡通渲染是非真实化渲染(NPR)的一个部分.而NPR旨在模拟出手工插图的效果例如油画、墨水画、漫画风格作品。 …

【Vulnhub 靶场】【hacksudo: FOG】【简单 - 中等】【20210514】

1、环境介绍 靶场介绍:https://www.vulnhub.com/entry/hacksudo-fog,697/ 靶场下载:https://download.vulnhub.com/hacksudo/hacksudo-FOG.zip 靶场难度:简单 - 中等 发布日期:2021年05月14日 文件大小:1.3 GB 靶场作…

禅道v11.6 基于linux环境下的docker容器搭建的靶场

一、环境搭建 linux环境下的 在docker环境下安装禅道CMS V11.6 docker run --name zentao_v11.6 -p 8084:80 -v /u01/zentao/www:/app/zentaopms -v /u01/zentao/data:/var/lib/mysql -e MYSQL_ROOT_PASSWORD123456 -d docker.io/yunwisdom/zentao:v11.6二、常见问题 1.删除…

【数电笔记】16-卡诺图绘制(逻辑函数的卡诺图化简)

目录 说明: 最小项卡诺图的组成 1. 相邻最小项 2. 卡诺图的组成 2.1 二变量卡诺图 2.2 三表变量卡诺图 2.3 四变量卡诺图 3. 卡诺图中的相邻项(几何相邻) 说明: 笔记配套视频来源:B站;本系列笔记并…

通义千问开源了 720 亿、70亿、140亿、Qwen-VL 四个大模型:实现“全尺寸、全模态”开源

本心、输入输出、结果 文章目录 通义千问开源了 720 亿、70亿、140亿、Qwen-VL 四个大模型:实现“全尺寸、全模态”开源前言阿里云CTO周靖人阿里云72B 的通义千问性能如何Qwen-1.8B花有重开日,人无再少年实践是检验真理的唯一标准 通义千问开源了 720 亿…

NSS [HUBUCTF 2022 新生赛]Calculate

NSS [HUBUCTF 2022 新生赛]Calculate 题目描述:python is a good tool in CTF 需要答对20题,每题回答时间(其实就是两次发包之前的间隔)要大于一秒小于三秒。 抓个包,我们的答案是POST发包。并且在这里看到了cookie&…

进程(5)——进程终止【linux】

进程 (4)——进程终止【linux】 一. 进程结束情况i. 正常终止ii. 出错终止iii. 异常退出 二. 进程返回值(针对正常和出错)2.1. 进程的退出方式i. returnii. exitiii. _exit 2.2. 查看C语言中的对应返回值的对应出错2.3 使用errno2…

华为变革进展指数TPM的五​个级别:试点级、推行级、功能级、集成级和世界级

华为变革进展指数TPM的五​个级别:试点级、推行级、功能级、集成级和世界级 TPM(Transformation Progress Metrics,变革进展指标)用来衡量管理体系在华为的推行程度和推行效果,并找出推行方面的不足与问题,…

数据结构(超详细讲解!!)第二十六节 图(中)

1.存储结构 1.邻接矩阵 图的邻接矩阵表示法(Adjacency Matrix)也称作数组表示法。它采用两个数组来表示图: 一个是用于存储顶点信息的一维数组;另一个是用于存储图中顶点之间关联关系的二维数组,这个关联关系数组被…

数据结构第二次作业——递归、树、图【考点罗列//错题正解//题目解析】

目录 一、选择题 ——递归—— 1.【单选题】 ——递归的相关知识点 2.【单选题】——递归的应用 3.【单选题】——递归的实现结构 4.【单选题】——递归的执行与实现 5.【单选题】 ——递归算法 ——树—— 6.【单选题】 ——树的结构 *7.【单选题】——树的知识点 …

STM32下载程序的五种方法

刚开始学习 STM32 的时候,很多小伙伴满怀热情买好了各种设备,但很快就遇到了第一个拦路虎——如何将写好的代码烧进去这个黑乎乎的芯片~ STM32 的烧录方式多样且灵活,可以根据实际需求选择适合的方式来将程序烧录到芯片中。本文将…