Hibernate sequence生成策略生成的id怎么会出现负数?

逗比驾到 2016-02-21 06:54:56
我使用的是Oracle 11g R2数据库,我想对一张user表的id列自动生成主键,用的是序列,sql原始代码如下:

/* Oracle 11g 数据库脚本 */

CREATE TABLE t_user(
id NUMBER(10) PRIMARY KEY,
name VARCHAR(20)
);

/* 创建序列 */
CREATE SEQUENCE user_seq;




建立实体类

package nuc.sw.entity;

import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.SequenceGenerator;
import javax.persistence.Table;
@Entity
@Table(name="t_user")
@SequenceGenerator(sequenceName="user_seq",name="user_seq")
public class User {
private int id;
private String name;
@Id
@GeneratedValue(strategy=GenerationType.SEQUENCE,generator="user_seq")
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}

}




接下来我写了一段这样的代码,目的是向数据库中插入一条记录(我整合了Spring4.2.4):

package nuc.sw.dao.impl;

import javax.annotation.Resource;

import org.springframework.orm.hibernate5.HibernateTemplate;
import org.springframework.stereotype.Component;

import nuc.sw.dao.UserDAO;
import nuc.sw.entity.User;
@Component("userDAOImpl")
public class UserDAOImpl implements UserDAO{
private HibernateTemplate hibernateTemplate;

public HibernateTemplate getHibernateTemplate() {
return hibernateTemplate;
}

@Resource
public void setHibernateTemplate(HibernateTemplate hibernateTemplate) {
this.hibernateTemplate = hibernateTemplate;
}

@Override
public void save(User user) throws Exception{
hibernateTemplate.save(user);
System.out.println("user "+user.getName()+" saved!");
}

}





package nuc.sw.service;

import javax.annotation.Resource;

import org.springframework.stereotype.Component;

import nuc.sw.dao.UserDAO;
import nuc.sw.dao.impl.UserDAOImpl;
import nuc.sw.entity.User;
@Component("userService")
public class UserService {

private UserDAO userDAO;

public UserDAO getUserDAO() {
return userDAO;
}

@Resource(name="userDAOImpl")
public void setUserDAO(UserDAO userDAO) {
this.userDAO = userDAO;
}

public void addUser(User user) throws Exception {
userDAO.save(user);
}
}





编写junit测试文件:

package nuc.sw.service;

import static org.junit.Assert.*;
import nuc.sw.entity.User;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class UserServiceTest {
@Test
public void testAddUser() throws Exception {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService service = (UserService)ctx.getBean("userService");
User user = new User();
user.setName("张三");
service.addUser(user);
assertTrue(true);
}

}





Spring的配置文件:

<?xml version="1.0" encoding="utf-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.2.xsd
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.2.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.2.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-4.2.xsd">

<context:annotation-config/>
<context:component-scan base-package="nuc.sw"/>

<bean class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<value>classpath:jdbc.properties</value>
</property>
</bean>

<bean id="dataSource" destroy-method="close"
class="org.apache.commons.dbcp2.BasicDataSource">
<property name="driverClassName" value="${jdbc.driverClassName}"/>
<property name="url" value="${jdbc.url}"/>
<property name="username" value="${jdbc.username}"/>
<property name="password" value="${jdbc.password}"/>
</bean>

<bean id="sessionFactory" class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<property name="packagesToScan">
<list>
<value>nuc.sw.entity</value>
</list>
</property>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">
org.hibernate.dialect.Oracle10gDialect
</prop>
<prop key="hibernate.show_sql">true</prop>
<prop key="hibernate.format_sql">true</prop>
</props>
</property>
</bean>

<bean id="hibernateTemplete"
class="org.springframework.orm.hibernate5.HibernateTemplate">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>

<bean id="txManager" class="org.springframework.orm.hibernate5.HibernateTransactionManager">
<property name="sessionFactory" ref="sessionFactory"/>
</bean>

<aop:config>
<aop:pointcut expression="execution(public * nuc.sw.service..*.*(..))"
id="businessService"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="businessService"/>
</aop:config>

<tx:advice id="txAdvice" transaction-manager="txManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
</beans>

程序运行没有出现任何异常,但是结果id继1之后却是从-46开始插一条+1,真是奇怪了:

...全文
536 5 打赏 收藏 转发到动态 举报
写回复
用AI写文章
5 条回复
切换为时间正序
请发表友善的回复…
发表回复
逗比驾到 2017-08-15
  • 打赏
  • 举报
回复
总结:其实还是自己的惯性思维惹的麻烦,总觉得Oracle里面已经设置了步长和初始值Hibernate就可以读到了,就不用设置了,总觉得Hibernate应该按照Oracle数据库的已有默认值来运作,但事实上并非如此。以后用到序列要注意了!
逗比驾到 2017-08-15
  • 打赏
  • 举报
回复
谢谢大家,问题已经解决。z634075513说的很对,问题出在了Hibernate的注解上少设置了属性。 学过Oracle的人都知道Oracle数据库11gR2版本用SQL创建序列如果不设定步长默认是1,不设定初始值默认是1。我创建序列是自己手工写代码创建的,没有指定这些值,但我在数据库GUI工具上取序列的值是没问题的。 但是一到了hibernate去取值的时候就不对了。我当时猜想是不是必须要Hibernate创建的序列就没问题呢,不能我自己手工创建序列吗?我删掉数据库的序列,让Hibernate去创建序列,这个时候序列的值就正常了。 我觉得Hibernate不可能这么设计,肯定是哪里设置不合适了。我是用注解编程,所以注意力转移到了@SequenceGenerator 这个注解上,经过一番倒腾发现是因为缺少了initialValue=1这个属性值,allocationSize属性是用来指定步长的,initialValue指定了序列的初始值。 修改后的代码应该是这样的: @Entity @Table(name="t_user") @SequenceGenerator(sequenceName="user_seq",name="user_seq",initialValue=1) @Component("user") public class User { private int id; private String name; @Id @GeneratedValue(strategy=GenerationType.SEQUENCE,generator="user_seq") public int getId() { return id; } 后来我发现,即便不指定allocationSize也可以得出正确结果,但是不指定initialValue=1结果就不对了。 所以提醒使用Hibernate 5的开发人员,为了防止序列产生不符合预期的值,使用@SequenceGenerator注解时一定要同时指定四个值:sequenceName、name、initialValue=1和allocationSize=1,这样就可以解决问题了。
z634075513 2016-06-01
  • 打赏
  • 举报
回复
我也遇到你这个问题看源码后解决了 http://blog.csdn.net/z634075513/article/details/51554194
zjs91 2016-02-22
  • 打赏
  • 举报
回复
CREATE SEQUENCE emp_sequence --序列名 INCREMENT BY 1 -- 每次加几个 START WITH 1 -- 从1开始计数 NOMAXVALUE -- 不设置最大值 NOCYCLE -- 一直累加,不循环 CACHE 10; 创建序列,指明一些参数
逗比驾到 2016-02-21
  • 打赏
  • 举报
回复
而且更让人感到怪异的是,如果让hibernate自己生成sequence,就不会有这样的问题,但如果是用我自己创建的表和sequence就不可以,我不明白我自己建的表和hibernate建的表一样一样的,输出的建表语句都一样,怎么会出这种奇葩问题?不过所幸的是,不影响程序的执行

67,512

社区成员

发帖
与我相关
我的任务
社区描述
J2EE只是Java企业应用。我们需要一个跨J2SE/WEB/EJB的微容器,保护我们的业务核心组件(中间件),以延续它的生命力,而不是依赖J2SE/J2EE版本。
社区管理员
  • Java EE
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧