项目中用hibernate容易出现错误的几个地方小记一下

平时做项目用到hibernate的时候,总是有一些小的地方容易出错,解决之后如果隔一段时间不用的话又容易忘记,所以特此小记下一,免得遇到问题又到处找资料...

1、一张表有一个int类型的自动增长字段(非主键),同时有一个varchar类型的字符串作为主键,hibernate进行ORM对象关系映射之后,会默认为如下所述:


如果什么都不作改动的话,不能进行添加等操作,会报如下错误:

not-null property references a null or transient value: com.eok.entity.MoveHeaders.moveHeaderId;nested exception is org.hibernate.PropertyValueException: not-null property references a null or transient value: com.eok.entity.MoveHeaders.moveHeaderId
进行如下改动之后就ok了,如:
之前:	<id name="moveNumber" type="java.lang.String">
            <column name="move_number" length="30" />
            <generator class="native" />
        </id>
        <property name="moveHeaderId" type="java.lang.Integer">
            <column name="move_header_id" not-null="true" unique="true">
                <comment>调拨单内部编号。</comment>
            </column>
        </property>
之后:<id name="moveNumber" type="java.lang.String">
            <column name="move_number" length="30" />
            <generator class="assigned" />------->把native改为assigned自行管理即可
        </id>
        <property name="moveHeaderId" type="java.lang.Integer" update="false" insert="false">----->把update和insert都设置为false
            <column name="move_header_id" not-null="true" unique="true">
                <comment>调拨单内部编号。</comment>
            </column>
        </property>
2、如果一张表中的一个字段是唯一约束的,我们在添加或修改之前会判断一下是否已经添加过相同的记录,这样的话就会先去查询,如果不存在相同记录的话再执行添加,这里就会出现一个共用session的问题,会报如下错误:
Batch update returned unexpected row count from update [0]; actual row count: 0; expected: 1
解决方案:在对应的dao层里面调用update()方法的时候加一个merger()强制关联一下就好了,如:
改之前:@Override
	public boolean updateMoveHeaders(MoveHeaders move) {
		return super.updateObject(move);
	}
改之后:@Override
	public boolean updateMoveHeaders(MoveHeaders move) {
		return super.updateObject(super.merge(move));
	}

3、如果有两张表A和B,B中有一个外键去关联A中的一个字符串,而这个字符串又非主键,这个时候hibernate默认会把其当成int类处理,你在B表做查询的时候,如果把对应的lazy设置为false的话,就会把对应的A表中的数据一起查询出来,这个时候就会报如下错误:
hibernate Invalid value for getInt() - '方案名称1'
解决方案:在<many-to-one>中加property-ref属性直接关联,如下:
改之前:<many-to-one name="inventorySolution" class="com.eok.entity.InventorySolution" fetch="select" lazy="false">
            <column name="inventory_solution" not-null="true">
                <comment>盘点方案的名称</comment>
            </column>
        </many-to-one>
改之后:<many-to-one name="inventorySolution" class="com.eok.entity.InventorySolution" fetch="select" lazy="false" property-ref="inventorySolutionName">
            <column name="inventory_solution" not-null="true">
                <comment>盘点方案的名称</comment>
            </column>
        </many-to-one>

4、通过get和load取查数据的时候,第二个参数一般传的是主键列对应的实体属性值,可为int也可以是字符串。load一般可以用在延迟加载上,使用一般保证能够查询到值时才用load,且用它查出来的值要在实际使用的时候才进行真正的操作。如:
Student stu = super.load(Student.class,1);-------->未操作数据库
System.out.println(stu.getName());---------->使用其的时候才真正去操作数据库
而get相反
Student stu = super.get(Student.class,1);-------->直接操作数据库

5、事务一般都放在service层中去处理,而非dao层,因为service才是真正处理业务逻辑的,而dao只是承担访问数据或者说成持久化数据的责任而已。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值