`

spring整合hibernate

阅读更多

原文转自:http://wanqiufeng.blog.51cto.com/409430/484739

(1)
简介:
在过去几年里,Hibernate不断发展,几乎成为Java数据库持久性的事实标准。它非常强大、灵活,而且具备了优异的性能。在本文中,我们将了解如何使用Java 5 注释来简化Hibernate代码,并使持久层的编码过程变得更为轻松。
传统上,Hibernate的配置依赖于外部 XML 文件:数据库映射被定义为一组 XML 映射文件,并且在启动时进行加载。
   
在最近发布的几个Hibernate版本中,出现了一种基于 Java 5 注释的更为巧妙的新方法。借助新的 Hibernate Annotation 库,即可一次性地分配所有旧映射文件——一切都会按照您的想法来定义——注释直接嵌入到您的Java 类中,并提供一种强大及灵活的方法来声明持久性映射。

即利用hibernate注解后,可不用定义持久化类对应的*.hbm.xml文件,直接以注解方式写入在持久化类中来实现。

Hibernate annotation使用了ejb JPA的注解,所以,下面安装配置hibernate annotation环境时,需要导入ejb的包。许多网上的资料都是jpa hibernate annotation方面的资料。

(2)
安装 Hibernate Annotation
第一步,
环境与jar包:
要使用 Hibernate Annotation,您至少需要具备 Hibernate 3.2Java 5。可以从 Hibernate 站点下载 Hibernate 3.2 Hibernate Annotation库。除了标准的 Hibernate JAR 和依赖项之外,您还需要 Hibernate Annotations .jar 文件(hibernate-annotations.jar)、Java 持久性 API lib/ejb3-persistence.jar)。

添加hibernate3.2.jar,hibernate-annotations- 3.3.0.jarhibernate-commons-annotations.jarejb3-persistence.jar 。这样就可以使用hibernateannotation了。

如果您正在使用 Maven,只需要向 POM 文件添加相应的依赖项即可,如下所示:
    ...
    <dependency>
      <groupId>org.hibernate</groupId>
      <artifactId>hibernate</artifactId>
      <version>3.2.1.ga</version>
    </dependency>
    <dependency>
      <groupId>org.hibernate</groupId>
      <artifactId>hibernate-annotations</artifactId>
      <version>3.2.0.ga</version>
    </dependency>
    <dependency>
      <groupId>javax.persistence</groupId>
      <artifactId>persistence-api</artifactId>
      <version>1.0</version>
    </dependency>

第二步,
获取 Hibernate 会话工厂。尽管无需惊天的修改,但这一工作与使用 Hibernate Annotations有所不同。您需要使用 AnnotationConfiguration 类来建立会话工厂:
sessionFactory = new AnnotationConfiguration().buildSessionFactory(); 
第三步,
尽管通常使用 <mapping> 元素来声明持久性类,您还是需要在 Hibernate 配置文件(通常是 hibernate.cfg.xml)中声明持久性类:
<!DOCTYPE hibernate-configuration PUBLIC
    "-//Hibernate/Hibernate Configuration DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
        <hibernate-configuration>
          <session-factory>
            <mapping class="com.onjava.modelplanes.domain.PlaneType"/>
            <mapping class="com.onjava.modelplanes.domain.ModelPlane"/>
          </session-factory>
        </hibernate-configuration>
近期的许多 Java 项目都使用了轻量级的应用框架,例如 Spring。如果您正在使用 Spring 框架,可以使用

AnnotationSessionFactoryBean 类轻松建立一个基于注释的 Hibernate 会话工厂,如下所示:
<!-- Hibernate session factory -->
  <bean id="sessionFactory"
       class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
   <property name="dataSource">
     <ref bean="dataSource"/>
   </property>
   <property name="hibernateProperties">
     <props>
       <prop key="hibernate.dialect">org.hibernate.dialect.DerbyDialect</prop>
       <prop key="hibernate.hbm2ddl.auto">create</prop>
       ...
     </props>
   </property>

 

   <!--指明使用标注的实体类-->
   <property name="annotatedClasses">
     <list>
       <value>com.onjava.modelplanes.domain.PlaneType</value>
       <value>com.onjava.modelplanes.domain.ModelPlane</value>

       ...
     </list>
   </property>

 

   <!--当然也可以不使用上面这种指定的方式,而使用包扫描的方式做为替换,推荐这种-->

    <property name="packagesToScan">
       <list>
            <value>
com.onjava.modelplanes.domain.*</value>
        </list>
      </property>
</bean>

(3)
hibernate Annotation
标签的使用:

[1]
1.
带注释的持久性类也是普通 POJO,它们只是具备了持久性注释的普通 POJO

2.事实上,您既可以保持字段的持久性(注释写在成员变量之上),也可以保持属性(注释写在getter方法之上)的持久性。

3.常用的hibernate annotation标签如下:

@Entity              --注释声明该类为持久类。将一个Javabean类声明为一个实体的数据库表映射类,最好实现序列化.此时,默认情况下,所有的类属性都为映射到数据表的持久性字段.若在类中,添加另外属性,而非映射来数据库的, 要用下面的Transient来注解.


@Table(name=
"promotion_info")      --持久性映射的表(表名="promotion_info).@Table是类一级的注解,定义在@Entity,为实体bean映射表,目录和schema的名字,默认为实体bean的类名,不带包名.


@Id--
注释可以表明哪种属性是该类中的独特标识符(即相当于数据表的主键) 
@GeneratedValue   --
定义自动增长的主键的生成策略
@Transient             --
将忽略这些字段和属性,不用持久化到数据库.适用于,在当前的持久类中,某些属性不是用于映射到数据表,而是用于其它的业务逻辑需要,这时,须将这些属性进行transient的注解.否则系统会因映射不到数据表相应字段而出错
@Temporal(TemporalType.TIMESTAMP)--
声明时间格式 
@Enumerated         --
声明枚举 
@Version                --
声明添加对乐观锁定的支持 
@OneToOne            --
可以建立实体bean之间的一对一的关联 
@OneToMany          --
可以建立实体bean之间的一对多的关联 
@ManyToOne          --
可以建立实体bean之间的多对一的关联 
@ManyToMany        --
可以建立实体bean之间的多对多的关联 
@Formula               --
一个SQL表达式,这种属性是只读的,不在数据库生成属性(可以使用sumaveragemax
@OrderBy               --Many
端某个字段排序(List)

 

1.2
Hibernate
能够出色地自动生成主键。Hibernate/EBJ 3 注释也可以为主键的自动生成提供丰富的支持,允许实现各种策略。
其生成规则由@GeneratedValue设定的.这里的@id@GeneratedValue都是JPA的标准用法, JPA提供四种标准用法,@GeneratedValue的源代码可以明显看出
JPA
提供的四种标准用法为TABLE,SEQUENCE,IDENTITY,AUTO. 
TABLE
:使用一个特定的数据库表格来保存主键。 
SEQUENCE
:根据底层数据库的序列来生成主键,条件是数据库支持序列。 
IDENTITY
:主键由数据库自动生成(主要是自动增长型) 
AUTO
:主键由程序控制。

在指定主键时,如果不指定主键生成策略,默认为AUTO 
@Id

相当于

@Id
@GeneratedValue(strategy = GenerationType.AUTO)

 

identity:

使用SQL Server MySQL 的自增字段,这个方法不能放到 Oracle 中,Oracle 不支持自增字段,要设定sequenceMySQL SQL Server 中很常用)。

Oracle就要采用sequence.

 

同时,也可采用uuid,native等其它策略.(相关用法,上网查询)

[2]

第一个持久性类 

@Entity 
@Table(name=
"T_MODEL_PLANE"
public class ModelPlane    implements Serializable { 
        @Id 
        @Column(name=
"PLANE_ID")    
        @GeneratedValue(strategy=GenerationType.AUTO) 
//注解于属性中 
/* 
对于oracle想使用各自的Sequence,设置如下:        
@GeneratedValue(strategy = GenerationType.AUTO,generator="PROMOTION_SEQ")        
@SequenceGenerator(name="PROMOTION_SEQ",sequenceName="PROMOTION_SEQ")    

 

另外:

对于自动增长后,在数据表中的相应字段,要设置字段为auto_increment.
*/
 
        
private Long id; 

        
private String name;//注解写于getter方法之上.请见下

     
//DATE            - java.sql.Date        
     
//TIME            - java.sql.Time        
     
//TIMESTAMP - java.sql.Timestamp        
     @Temporal(TemporalType.TIMESTAMP)        
     @Column(name=
"start_time")        
     
private Date startTime;     

     
//显示0 隐藏1        
     
public static enum DisplayType {显示,隐藏}        
     @Enumerated(value = EnumType.ORDINAL)
//ORDINAL序数        
     
private DisplayType displayType = DisplayType.显示;    

        
//1.sql语句中的字段和表名都应该和数据库相应,而不是类中的字段,        
     
//若带有参数如la.id= id,这个=id才是类中属性        
     
//2.操作字段一定要用别名        
     @Formula(select COUNT(la.id) from largess la)        
     
private int count;    

        
//注解于方法中 
        @Column(name=
"PLANE_ID", length=80, nullable=true//较详细定义 
        
public String getName() { 
                
return name; 
        } 
        
public void setName(String name) { 
                
this.name = name; 
        } 
其它的setter,getter省略...... 
}


该内容将映射到下表中: 
CREATE TABLE T_MODEL_PLANE 

        PLANE_ID 
long
        PLANE_NAME varchar 
        
其它字段省略... 
)    

默认情况下,Hibernate 会将持久类以匹配的名称映射到表和字段中。例如,下例中,若不用注解,则会映射到如下一表中:
CREATE TABLE MODELPLANE
(
    ID long,
    NAME varchar

    其它字段省略...
)

[3]
一对多注解:

1.

在一对多注解中,会用到:

""方:
@OneToMany --> mappedBy:"
"方的关联属性(被控方)
"
"方:
@ManyToOne --> @JoinColumn
""方定义的外键字段.

如数据表定义外键如下:

FOREIGN KEY (classid) REFERENCES classes(id)

:

@JoinColumn(name="classid"

2.

在双向关联中,有且仅有一端作为主体(owner)端存在:主体端负责维护联接列(即更新),对于不需要维护这种关系的从表则通过mappedNy属性进行声明。mappedBy的值指向另一主体的关联属性。例子中,mappedBy的值为classes

附加说明:

mappedBy相当于过去的inverse="true".
inverse=false
sideside其实是指inversefalse所位于的class元素)端有责任维护关系,而inversetrue端无须维护这些关系。

3.

cascadefetch使用说明:

Cascade

CascadeType.PERSIST (级联新建) 
CascadeType.REMOVE 
(级联删除) 
CascadeType.REFRESH
(级联刷新) 
CascadeType.MERGE  
(级联更新)中选择一个或多个。 
CascadeType.ALL

 fetch属性:

关联关系获取方式,即是否采用延时加载。
 LAZY
(默认值)采用延时加载,查询数据时,不一起查询关联对象的数据。而是当访问关联对象时(如:getStudnets())才触发相应的查询操作,获取关联对象数据。
EAGER
:是在查询数据时,也直接一起获取关联对象的数据。

package oneToMany; 
import java.util.Set; 
import javax.persistence.*; 
/* 
注意导入时,是导入:import javax.persistence.*;    
非导入org.hibernate的相关类:import org.hibernate.annotations.Entity; 
*/
 
@Entity 
@Table(name=
"classes"
public class Classes implements Serializable { 
  @Id 
  @GeneratedValue(strategy=GenerationType.AUTO) 
  
private int id; 
  
private String name; 
    
  @OneToMany(cascade=CascadeType.ALL,mappedBy=
"classes")    
  
private Set<Student> students; 
//getter,setter省略 



package oneToMany; 
import javax.persistence.*; 
@Entity 
@Table(name=
"student"
public class Student implements Serializable  { 
  @Id 
  @GeneratedValue(strategy=GenerationType.AUTO) 
  
private int sid; 
    
  
private String sname; 
    
  
//若有多个cascade,可以是:{CascadeType.PERSIST,CascadeType.MERGE} 
  @ManyToOne(cascade={CascadeType.ALL})         
  @JoinColumn(name=
"classid")     //student类中对应外键的属性:classid 
  
private Classes classes; 
//getter,setter省略 



public class TestOneToMany { 
/* 
CREATE TABLE    student (    --
要定义外键!!!!!!! 
    `sid` double NOT NULL auto_increment, 
    `classid` double NULL, 
    `sname` varchar(255) NOT NULL, 
    PRIMARY KEY    (sid), 
    INDEX par_ind (classid), 
    FOREIGN KEY (classid) REFERENCES classes(id) ON DELETE CASCADE ON UPDATE CASCADE 
) ENGINE=InnoDB 
*/
    
  
public static void main(String[] args) throws SQLException    
  { 
    
try 
    { 
      SessionFactory sf = 
new AnnotationConfiguration().configure().buildSessionFactory(); 
      Session session=sf.openSession(); 
      Transaction tx=session.beginTransaction();         

/*

因为mappedBy是定义在classes,classes类不负责维护级联关系.即维护者是student.所以,

1.要将clsses的数据,赋给student,即用studentsetClasses()方法去捆定class数据;

2.在进行数据插入/更新session.save()/session.update(),最后操作的是student.

*/
      Classes classes=
new Classes(); 
      classes.setName(
"access"); 
        
      Student st1=
new Student(); 
      st1.setSname(
"jason"); 
      st1.setClasses(classes); 
      session.save(st1); 
        
      Student st2=
new Student(); 
      st2.setSname(
"hwj"); 
      st2.setClasses(classes); 
      session.save(st2); 
      tx.commit();

/* 

输出如下:

Hibernate: insert into classes (name) values (?)
Hibernate: insert into student (classid, sname) values (?, ?)
Hibernate: insert into student (classid, sname) values (?, ?)

*/

/*

因为一端维护关系另一端不维护关系的原因,我们必须注意避免在应用中用不维护关系的类(class)建立关系,因为这样建立的关系是不会在数据库中存储的。

如上的代码倒过来,则插入时,student的外键值为空.如下:

*/

//      Student st1=new Student(); 
//      st1.setSname("jason"); 

//      session.save(st1); 
//        
//      Student st2=new Student(); 
//      st2.setSname("hwj"); 
//      session.save(st2); 
//        
//      Set<Student> students=new HashSet<Student>(); 
//      students.add(st1); 
//      students.add(st2); 
//        
//      Classes classes=new Classes(); 
//      classes.setName("access"); 
//      classes.setStudents(students); 
//      session.save(classes); 

/*

输出如下:
Hibernate: insert into student (classid, sname) values (?, ?)
Hibernate: insert into student (classid, sname) values (?, ?)

Hibernate: insert into classes (name) values (?)

*/
    } 
    
catch(HibernateException e) 
    { 
      e.printStackTrace();        
    } 
  } 
}

 

 

[4]
多对多注解

在多对多注解中,双方都采用@ManyToMany.

其中被控方,像一对多注解中设置一样,也要设置mappedBy.

其中主控方,不像一对多注解那样,采用@joinColumn,而是采用@joinTable.如下:

@JoinTable(name="j_student_course" ,joinColumns={@JoinColumn(name="sid")},inverseJoinColumns={@JoinColumn(name="cid")})

其中,

如上所说,mappedBy,相当于inverse="true".所以,@joinTable中的inverseJoinColumns中定义的字段为mappedBy所在类的主键.
joinColumns
定义的字段,就是当前类的主键.

@Entity 
@Table(name=
"jcourse"
public class Jcourse { 
  @Id 
  @GeneratedValue(strategy=GenerationType.AUTO) 
  
private int cid; 
  
private String cname; 
    
  @ManyToMany(cascade={CascadeType.PERSIST,CascadeType.MERGE},fetch=FetchType.LAZY ,mappedBy=
"courses"
  
private Set<Jstudent> students; 
//setter,getter省略....    



@Entity 
@Table(name=
"jstudent"
public class Jstudent { 
  @Id 
  @GeneratedValue(strategy=GenerationType.AUTO) 
  
private int sid; 
    
  
private String sname; 
    
  @ManyToMany(cascade={CascadeType.PERSIST,CascadeType.MERGE},fetch=FetchType.EAGER) 
  
//inverseJoinColumns中对应的id为以下属性course的对应id. 
  @JoinTable(name=
"j_student_course" ,joinColumns={@JoinColumn(name="sid")},inverseJoinColumns={@JoinColumn(name="cid")}) 
  
private Set<Jcourse> courses; 
//setter,getter省略....    



public class Test { 
  
public static void main(String[] args) { 
    
try 
    { 
      SessionFactory sf = 
new AnnotationConfiguration().configure().buildSessionFactory(); 
      Session session=sf.openSession(); 
      Transaction tx=session.beginTransaction(); 
        
      Jcourse course=
new Jcourse(); 
      course.setCname(
"jason-english"); 
      session.save(course); 
//先各自保存
        
      Jcourse course2=
new Jcourse(); 
      course2.setCname(
"herry-english"); 
      session.save(course2); 
        
      Set<Jcourse> courses=
new HashSet<Jcourse>(); 
      courses.add(course); 
      courses.add(course2); 
        
      Jstudent student=
new Jstudent(); 
      student.setSname(
"jason"); 
      student.setCourses(courses); 
        
      session.save(student);
// 要用非mapby定义的类(studet)来作为主者(会控制级联关系),一对多,多对一也一样道理
      
//可以尝试反过来
      tx.commit(); 
    } 
    
catch(HibernateException e) 
    { 
      e.printStackTrace();        
    } 
  } 
}

 

 

分享到:
评论

相关推荐

    spring整合hibernate示例代码

    标题"spring整合hibernate示例代码"提示我们,我们将讨论如何在实际项目中结合这两个框架。Spring可以帮助管理Hibernate的SessionFactory和Transaction,提供声明式事务管理,以及通过AOP(面向切面编程)实现更灵活...

    Spring整合Hibernate.jar

    标题"Spring整合Hibernate.jar"意味着我们将讨论如何将这两个强大的框架集成在一起,以便在Spring管理的环境中使用Hibernate进行数据库操作。这通常涉及到以下步骤和知识点: 1. **引入依赖**:首先,你需要在项目...

    spring整合hibernate实现事务处理

    本篇文章将详细探讨如何通过Spring整合Hibernate来实现事务处理,重点介绍注解方式和XML配置方式。 首先,我们了解事务处理在数据库操作中的重要性。事务是一组操作,这些操作要么全部成功,要么全部失败,确保数据...

    spring整合hibernate的jar包

    标题中的“spring整合hibernate的jar包”指的是在Java Web开发中,Spring框架与Hibernate持久层框架的集成。这两个框架结合使用可以提供强大的数据访问和业务逻辑处理能力。Spring作为一个全面的轻量级框架,提供了...

    spring整合hibernate所用相关jar包

    本文将深入探讨Spring整合Hibernate的相关知识点,适合初学者入门。 首先,我们需要理解Spring的核心理念。Spring框架提供了一个轻量级的容器,它能够管理应用程序中的对象,包括初始化、配置和依赖注入。AOP则是...

    Spring整合Hibernate

    Spring4整合Hibernate4实现用户购买图书和结账等操作,整合主要实现用IoC容器来管理Hibernate的SessionFactory实例,并使Hibernate使用Spring所提供的声明式事务……

    spring整合hibernate实例

    这篇名为"spring整合hibernate实例"的内容,显然是关于如何将这两个框架协同工作,构建一个高效、灵活的Java应用的教程。在整合过程中,我们将探讨以下几个关键知识点: 1. **Spring的ApplicationContext**: 这是...

    Spring整合HIbernate

    《Spring整合Hibernate实战指南》 在Java开发领域,Spring框架以其强大的依赖注入、AOP(面向切面编程)以及丰富的模块支持,成为了企业级应用开发的首选。而Hibernate作为持久层框架,以其对象关系映射(ORM)能力...

    Spring整合Hibernate 详解.doc

    Spring整合Hibernate是现代Java开发中常见的一种技术组合,利用Spring框架的强大功能来管理和协调Hibernate的持久化操作。Spring为Hibernate提供了全面的集成方案,简化了DAO(Data Access Object)的开发,同时也...

    使用spring整合hibernate和struts时所要用到的所有jar包

    首先,我们需要理解Spring如何与Hibernate和Struts进行整合: 1. **Spring与Hibernate整合**: - Spring通过其`HibernateTemplate`或`HibernateDaoSupport`类简化了对Hibernate的操作,提供了事务管理。你可以定义...

    Spring整合Hibernate示例完整代码

    下面,我们将深入探讨Spring整合Hibernate的相关知识点。 1. **依赖注入**:Spring框架的核心特性之一是依赖注入(DI),它允许我们在不进行硬编码的情况下管理对象之间的依赖关系。在整合Hibernate时,Spring可以...

    spring整合hibernate与struts2所需jar包

    标题"spring整合hibernate与struts2所需jar包"表明我们需要关注的是如何将这三个框架整合在一起,并且提供了所需的一些基础组件。整合SSH可以让开发者利用Spring的管理能力,让Hibernate更易于使用,同时通过Struts2...

    Spring整合Hibernate示例

    Spring整合Hibernate配置测试示例

    spring整合hibernate开发源码

    在"spring整合hibernate开发源码"的压缩包中,可能包含了以下内容: 1. **配置文件**:如`applicationContext.xml`,其中配置了Spring和Hibernate的相关bean,如DataSource、SessionFactory、TransactionManager等。...

    Spring整合Hibernate需要用到的部分jar包

    这里的jar包只是说Spring自带的关于Hibernate的jar包,即这四个jar包: hibernate3.jar, hibernate-annotations.jar, hibernate-entitymanager.jar hibernate-commons-annotations.jar 因为我原来下载了一个,但是...

    Spring+hibernate整合源代码

    这个“Spring+hibernate整合源代码”应该包含了实现上述整合步骤的示例代码,可以作为学习和参考的资源。通过学习和实践这些代码,你可以更好地理解和掌握 Spring 和 Hibernate 整合的细节,提升你的 Java Web 开发...

    spring整合Hibernate学习笔记.docx

    Spring 整合 Hibernate 是一种常见的企业级应用开发模式,它将 Spring 框架的管理优势与 Hibernate 的持久层功能结合起来,提供了更高效、更稳定的应用解决方案。在本学习笔记中,我们将深入探讨如何实现这一整合,...

    Spring整合hibernate(2)之基于HibernateTemplate的整合

    Spring整合Hibernate基于HibernateTemplate的方式,极大地简化了数据库操作,同时也让事务管理和代码的编写变得更加规范和高效。在实际项目中,可以根据需求进一步配置和优化,比如使用JPA的...

    spring整合hibernate例子

    当我们谈到"Spring整合Hibernate例子"时,这意味着我们将探讨如何将这两个强大的框架结合在一起,以实现更高效、更灵活的数据库操作。 Spring框架的核心特性之一是依赖注入(Dependency Injection,DI),这使得...

Global site tag (gtag.js) - Google Analytics