平时做项目用到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只是承担访问数据或者说成持久化数据的责任而已。