MySQL抛出不正确的字符串值错误



我正试图将下面的tweet存储到一个长文本列/utf8字符集/MQL5.5中。带有MyISAM存储的数据库。

我们还尝试了utf8mb4、utf16、utf32字符集,但无法解决此问题。

tweet="@Dorable_Dimples: Okay enough of those #IfYouWereMines I'm getting dep
ressed. #foreveralone ?" lol yes
mysql> ALTER DATABASE foo CHARACTER SET utf8 COLLATE utf8_bin;
mysql> show variables like 'char%';
+--------------------------+-------------------------------------------+
| Variable_name | Value |
+--------------------------+-------------------------------------------+
| character_set_client | utf8 |
| character_set_connection | utf8 |
| character_set_database | utf8 |
| character_set_filesystem | binary |
| character_set_results | utf8 |
| character_set_server | latin1 |
| character_set_system | utf8 |
| character_sets_dir | /rdsdbbin/mysql-5.5.12.R1/share/charsets/ |
Incorrect string value: 'xF0x9Fx98x94xE2x80...' for column 'tweet' at row 1
Unable to store tweet "@Dorable_Dimples: Okay enough of those #IfYouWereM
ines I'm getting depressed. #foreveralone ?" lol yes
javax.persistence.PersistenceException: org.hibernate.exception.GenericJDBCExcept
ion: could not insert
at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityMana
gerImpl.java:1387)
at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityMana
gerImpl.java:1315)
at org.hibernate.ejb.AbstractEntityManagerImpl.convert(AbstractEntityMana
gerImpl.java:1321)
at org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityMana
gerImpl.java:843)
at java.util.TimerThread.mainLoop(Timer.java:512)
at java.util.TimerThread.run(Timer.java:462)
at org.hibernate.exception.SQLStateConverter.handledNonSpecificException(
SQLStateConverter.java:140)
at org.hibernate.exception.SQLStateConverter.convert(SQLStateConverter.ja
va:128)
at org.hibernate.exception.JDBCExceptionHelper.convert(JDBCExceptionHelpe
r.java:66)
at org.hibernate.id.insert.AbstractReturningDelegate.performInsert(Abstra
ctReturningDelegate.java:64)
at org.hibernate.persister.entity.AbstractEntityPersister.insert(Abstract
EntityPersister.java:2345)
at org.hibernate.persister.entity.AbstractEntityPersister.insert(Abstract
EntityPersister.java:2852)
at org.hibernate.action.EntityIdentityInsertAction.execute(EntityIdentity
InsertAction.java:71)
at org.hibernate.engine.ActionQueue.execute(ActionQueue.java:273)
at org.hibernate.event.def.AbstractSaveEventListener.performSaveOrReplica
te(AbstractSaveEventListener.java:320)
at org.hibernate.event.def.AbstractSaveEventListener.performSave(Abstract
SaveEventListener.java:203)
at org.hibernate.event.def.AbstractSaveEventListener.saveWithGeneratedId(
AbstractSaveEventListener.java:129)
at org.hibernate.ejb.event.EJB3PersistEventListener.saveWithGeneratedId(E
JB3PersistEventListener.java:69)
at org.hibernate.event.def.DefaultPersistEventListener.entityIsTransient(
DefaultPersistEventListener.java:179)
at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultP
ersistEventListener.java:135)
at org.hibernate.event.def.DefaultPersistEventListener.onPersist(DefaultP
ersistEventListener.java:61)
at org.hibernate.impl.SessionImpl.firePersist(SessionImpl.java:808)
at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:782)
at org.hibernate.impl.SessionImpl.persist(SessionImpl.java:786)
at org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityMana
gerImpl.java:837)
... 5 more
Caused by: java.sql.SQLException: Incorrect string value: 'xF0x9Fx98x94xE2x
80...' for column 'tweet' at row 1
at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:1073)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3609)
at com.mysql.jdbc.MysqlIO.checkErrorPacket(MysqlIO.java:3541)
at com.mysql.jdbc.MysqlIO.sendCommand(MysqlIO.java:2002)
at com.mysql.jdbc.MysqlIO.sqlQueryDirect(MysqlIO.java:2163)
at com.mysql.jdbc.ConnectionImpl.execSQL(ConnectionImpl.java:2624)
at com.mysql.jdbc.PreparedStatement.executeInternal(PreparedStatement.jav
a:2127)
at com.mysql.jdbc.PreparedStatement.executeUpdate(PreparedStatement.java:
2427)
at com.mysql.jdbc.PreparedStatement.executeUpdate(PreparedStatement.java:
2345)
at com.mysql.jdbc.PreparedStatement.executeUpdate(PreparedStatement.java:
2330)
at org.hibernate.id.IdentityGenerator$GetGeneratedKeysDelegate.executeAnd
Extract(IdentityGenerator.java:94)
at org.hibernate.id.insert.AbstractReturningDelegate.performInsert(Abstra
ctReturningDelegate.java:57)

是推文末尾的字符导致了问题。

它看起来像一个"表情符号"字符,也就是日本的笑脸,但无论是Chrome还是Safari都不会显示给我。

在某些版本的MySQL中存储4字节utf字符存在已知问题。显然,您必须使用utf8mb4来表示4字节的UTF字符,因为普通的utf8字符集只能表示长度不超过3字节的字符,因此无法存储基本多语言平面之外的字符

http://dev.mysql.com/doc/refman/5.5/en/charset-unicode-utf8mb4.html

这对我来说是个新闻,因为这基本上意味着MySQL中的utf8数据类型不是真正合适的utf8。

这里有关于如何处理这一问题的建议如何在mysql中插入utf-8 mb4字符(ios5中的表情符号)?包括:

"还要确保你的应用层将其数据库连接的字符集设置为utf8mb4。仔细检查这是否真的发生了——如果你运行的是所选框架的mysql客户端库的旧版本,它可能没有使用utf8mb4支持进行编译,也不会正确设置字符集。如果没有,你可能必须更新它或自己编译它"

如果使用连接器/J,则需要在连接配置中设置character_set_server=utf8mb4。

你的所有角色集都应该是utf8mb4,你可能已经尝试过了,但目前还没有设置。

我喜欢Danask57的答案-这是正确的,也是正确的方法。(我自己投票支持)

然而,另一个快速而肮脏的解决方案是更改模式。使用varbinary或binary来存储tweet字符串:

http://dev.mysql.com/doc/refman/5.0/en/binary-varbinary.html

好处是你不会遇到任何字符集问题。

缺点是字符串比较和排序将丢失,并且无法对列进行全文索引。

这只是一个建议,但这不是"正确"的答案,只是一个快速而肮脏的解决方案。

我遇到了这个问题。要解决此问题,请按照以下优秀指南将mysql服务器端的默认编码更改为utf8mb4:http://mathiasbynens.be/notes/mysql-utf8mb4。

请记住,在对配置文件进行更改后重新启动mysqld服务。

对我来说,我还需要将mysqljdbc驱动程序更新到5.1.18版本(从5.1.6版本)。我在某个地方读到,您必须至少使用5.1.14版本才能很好地使用utf8mb4字符编码。希望这能有所帮助!

为什么在示例中引号之外有文本-即"lol yes"

tweet="@Dorable_Dimples: Okay enough of those #IfYouWereMines I'm getting depressed. #foreveralone ?" lol yes

问题出现在字符串"@"中。引擎数据库像一个特殊字符一样进行解释。我做:

   tweet="Dorable_Dimples: Okay enough of those #IfYouWereMines I'm getting dep

应力#总体而言?"lol yes

最新更新