开发避坑指南(29):微信昵称特殊字符存储异常修复方案

异常信息

java 复制代码
Cause: java.sql.SQLException: 
Incorrect string value: '\xF0\x9F\x8D\x8B\xE5\xBB...' for column 'nick_name' at row 1

异常背景

抽奖大转盘,抽奖后需要保存用户抽奖记录,用户再次进入游戏时根据抽奖记录判断剩余抽奖机会。保存抽奖记录时需要保存用户open_id,昵称,抽奖时间等信息,保存昵称时异常,没有成功记录抽奖导致剩余机会不减,用户可多次抽奖。

保存接口请求参数:

异常分析

昵称包含特殊字符,数据库定义的数据类型、字符集不支持,所以保存失败了。

解决办法

MySQL在5.5.3版本之后增加了这个utf8mb4的编码,mb4就是most bytes 4的意思,专门用来兼容四字节的unicode。低版本的MySQL

支持的utf8编码,最大字符长度为 3 字节,如果遇到 4 字节的字符就会出现错误了,其中Emoji表情是四个字节,导致插不进去。

解决这个办法可以从数据库层面上解决,即修改数据库的字段定义,如下方案1;或者从应用层面解决,如下方案2。

方案1:修改数据库字段字符集

mysql数据库中,将nick_name字段字符集由utf8修改为utf8mb4,数据库链接中将字符集由utf8改为utf8mb4。如下:

properties 复制代码
url: "jdbc:mysql://localhost:3306/big-turntable?useUnicode=true&characterEncoding=utf8mb4&useSSL=false"

查看日志和数据库,插入不再异常。

方案2:使用 Base64 编码与解码昵称

java 复制代码
//Base64编码保存昵称
String nickname = record.getNickName();
if(stringutil.isNotEmpty(nickname)){
    nickname = Base64.getEncoder().encodeTostring(nickname.getBytes( charsetName: "utf-8"));
    record.setNickName(nickname);
}   

//Base64解码昵称
if(stringutil.isNotEmpty(record.getNickName())){
    String nickName =record.getNickName();
    String nickName = new String(Base64.getDecoder().decode(nickName), 	 charsetName: "UTF-8");
    nickNameMap.put(record.getPriceId(),nickName);
}
相关推荐
期待のcode20 小时前
Spring框架1—Spring的IOC核心技术1
java·后端·spring·架构
a栋栋栋20 小时前
wsl 环境下用Docker 安装多版本MySQL
mysql·docker·容器
葵野寺20 小时前
【RelayMQ】基于 Java 实现轻量级消息队列(七)
java·开发语言·网络·rabbitmq·java-rabbitmq
书院门前细致的苹果21 小时前
JVM 全面详解:深入理解 Java 的核心运行机制
java·jvm
上官浩仁21 小时前
springboot excel 表格入门与实战
java·spring boot·excel
Hello.Reader1 天前
从零到一上手 Protocol Buffers用 C# 打造可演进的通讯录
java·linux·c#
树码小子1 天前
Java网络初识(4):网络数据通信的基本流程 -- 封装
java·网络
稻草人想看远方1 天前
GC垃圾回收
java·开发语言·jvm
en-route1 天前
如何在 Spring Boot 中指定不同的配置文件?
java·spring boot·后端
百锦再1 天前
在 CentOS 系统上实现定时执行 Python 邮件发送任务
java·linux·开发语言·人工智能·python·centos·pygame