java - Spring批量导入期间枚举到字符串的转换失败
问题描述
我正在使用 Spring Boot 和 Hiberna 创建 Spring 批处理作业,但在插入过程中遇到问题。这是代码的相关部分:
@Bean
public JdbcBatchItemWriter<OphthalmicLens> writer(DataSource dataSource) throws SQLException {
return new JdbcBatchItemWriterBuilder<OphthalmicLens>()
.itemSqlParameterSourceProvider(new BeanPropertyItemSqlParameterSourceProvider<>())
.sql("INSERT INTO OphthalmicLens (`createdBy`,`createdDate`,`lastModifiedBy`,`lastModifiedDate`,`sid`,`version`,`manufacturer`,`manufacturerCode`,`name`,`sku`,`upc`,`cylinder`,`design`,`diameter`,`index`,`material`,`source`,`sphere`,`type`) VALUES (:createdBy,NOW(),:lastModifiedBy,NOW(),UUID(),:version,:manufacturer,:manufacturerCode,:name,:sku,:upc,:cylinder,:design,:diameter,:index,:material,:source,:sphere,:type)")
.dataSource(dataSource).build();
}
属性设计是一个枚举:
public enum OphthalmicDesign {
SPHERIC, ASPHERIC, ATORIC, BIASPHERIC
}
在我的OphthalmicLens
豆子中使用的是这样的:
@NotNull
@Enumerated(EnumType.STRING)
@Column(nullable = false)
private OphthalmicDesign design = OphthalmicDesign.SPHERIC;
我正在使用 Mysql 5.7,并且数据库中的该列按预期映射为 VARCHAR(255)。
当我运行作业时,我收到此错误:
Caused by: java.sql.BatchUpdateException: Incorrect string value: '\xAC\xED\x00\x05~r...' for column 'design' at row 1
at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method) ~[?:1.8.0_172]
at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:62) ~[?:1.8.0_172]
at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45) ~[?:1.8.0_172]
at java.lang.reflect.Constructor.newInstance(Constructor.java:423) ~[?:1.8.0_172]
at com.mysql.cj.util.Util.handleNewInstance(Util.java:210) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.util.Util.getInstance(Util.java:185) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.util.Util.getInstance(Util.java:192) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.jdbc.exceptions.SQLError.createBatchUpdateException(SQLError.java:218) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.jdbc.ClientPreparedStatement.executeBatchedInserts(ClientPreparedStatement.java:768) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.jdbc.ClientPreparedStatement.executeBatchInternal(ClientPreparedStatement.java:444) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.mysql.cj.jdbc.StatementImpl.executeBatch(StatementImpl.java:839) ~[mysql-connector-java-8.0.11.jar:8.0.11]
at com.zaxxer.hikari.pool.ProxyStatement.executeBatch(ProxyStatement.java:128) ~[HikariCP-2.7.9.jar:?]
我了解问题是枚举未转换为字符串。我可以使用不同的方法来解决问题,BeanPropertyItemSqlParameterSourceProvider
但我认为这太过分了,而且我会失去 Spring 设施的优势。
即使我使用 Spring Batch,您是否有提示告诉 Spring/Hibernate 自动将该枚举转换为字符串?
解决方案
您可以指示BeanPropertyItemSqlParameterSourceProvider
返回VARCHAR
字段的类型,design
如下所示:
@Bean
public JdbcBatchItemWriter<OphthalmicLens> writer(DataSource dataSource) throws SQLException {
return new JdbcBatchItemWriterBuilder<OphthalmicLens>()
.itemSqlParameterSourceProvider(new BeanPropertyItemSqlParameterSourceProvider<OphthalmicLens>() {
@Override
public SqlParameterSource createSqlParameterSource(OphthalmicLens item) {
return new BeanPropertySqlParameterSource(item) {
@Override
public int getSqlType(String paramName) {
if (paramName.equalsIgnoreCase("desing")) {
return Types.VARCHAR;
}
return super.getSqlType(paramName);
}
};
}
})
.sql("INSERT INTO OphthalmicLens (`createdBy`,`createdDate`,`lastModifiedBy`,`lastModifiedDate`,`sid`,`version`,`manufacturer`,`manufacturerCode`,`name`,`sku`,`upc`,`cylinder`,`design`,`diameter`,`index`,`material`,`source`,`sphere`,`type`) VALUES (:createdBy,NOW(),:lastModifiedBy,NOW(),UUID(),:version,:manufacturer,:manufacturerCode,:name,:sku,:upc,:cylinder,:design,:diameter,:index,:material,:source,:sphere,:type)")
.dataSource(dataSource)
.build();
}
推荐阅读
- java - RabbitListener 在 Spring Boot 应用程序中一遍又一遍地使用相同的线程
- r - 从 R 中的 YYYYMM 列中提取月份
- kubernetes - 面向最终用户的 kubernaties 中的隔离应用程序
- deep-learning - 关于 caffe caffe.paramspec 没有名为压缩的参数
- redhat - 无法在 redhat 7.4 上注册订阅管理器 - “NoneType”对象没有属性“__getitem__”
- mysql - 根据表B中的年龄计算从表A中选择行
- android - 获取 imageView 颜色并设置补色
- tableau-api - Tableau 中的连接和数据源有什么区别?
- python - Scrapy:忽略响应 403
- angular - 如何显示离子列表中所选项目的实际计数?