首页 > 代码库 > Hibernate保存时报java.lang.ClassCastException

Hibernate保存时报java.lang.ClassCastException

  今天改需求的是遇到一个问题,Hibernate执行save()方法时报错:java.lang.ClassCastException: java.lang.String cannot be cast to java.math.BigDecimal。在网上搜了一下,几乎所有的回答都是说实体、映射文件和表结构之间类型不对应。检查了好几遍也没有发现问题,只好想其它办法实现,结果在写其它方法代码的时候发现了问题所在。

  我通过Hibernate去往一张视图的里存数据,视图是查通过dblink连接的一个数据库的单表,服务直接访问的数据库主键都是varchar()类型,通过dblikn访问的远程数据库主键是Number类型,而我在写映射文件的时候没有检查就把主键生成策略写成了UUID,于是在插入的时候生成的String值要插入number类型的主键就会报错。所以把主键生成策略改成identity或sequence就可以解决。

  修改过程中发现hibernate配置主键生成策略可以直接调用远程数据库的sequence。

<id name="XXXX" type="java.math.BigDecimal">
  <column name="XXXX" precision="XX" scale="0">
    <comment>XXXX</comment>
  </column>
  <generator class="sequence" >
    <param name="sequence">dblikn.XXXX</param>
  </generator>
</id>

  顺便记录一下前面提的另外一种解决办法,就是通过Hibernate直接执行sql语句。

String sql = "insert into XXXX(select (select sys_guid() from dual),XXXX from XXXX@dblink)";
super.getCurrentSession().createSQLQuery(sql).executeUpdate();

  没有写完,基本就是这几行代码。采用不同主键生成策略都要在代码中写出相应的主键。比如sequence要调用已有的sequence。另外Oracle不支持identity,还是头一次知道。

Hibernate保存时报java.lang.ClassCastException