Java Web系列:Hibernate 基礎

從如下5個方面學習hibernate ORM。html

(1)配置文件:hibernate.cfg.xml XML文件和hibernate.properties屬性文件java

(2)實體映射:1對多、多對多sql

(3)會話工廠與會話:SessionFactory&Session數據庫

(4)查詢:SQL原生查詢、HQL通用查詢、Criteria條件查詢apache

(5)事務:Transanctionsession

Hibernate的5個核心對象Conifguration、SessionFactory、Session、Query和Transanction是必須掌握的。另外,沒有相似Linq的語言集成查詢。併發

1.配置文件:hibernate.cfg.xml XML文件和hibernate.properties屬性文件

Hibernate使用Configuration表示配置信息,配置文件的信息最終會適配到Configuration對象。雖然XML文件一直被hibernate支持,但使用hibernate.properties屬性文件更簡潔。框架

HSQLDB數據庫是一個經常使用的JAVA版的測試數據庫,咱們經過下面兩種方式演示HSQLDB數據庫的配置。其中connection.driver_class, connection.url, connection.username 和 connection.password提供了JDBC使用的數據庫連接信息,dialect配置SQL方言,hbm2ddl.auto配置啓用自動更新數據庫模式,show_sql和format_sql配置便於咱們在控制檯查看輸出信息,generate_statistics配置生成統計信息。分佈式

(1)XML方式配置Hibernate:post

 1 <?xml version='1.0' encoding='utf-8'?>
 2 <!DOCTYPE hibernate-configuration PUBLIC
 3     "-//Hibernate/Hibernate Configuration DTD//EN"
 4     "http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
 5 
 6 <hibernate-configuration>
 7     <session-factory>
 8         <property name="hibernate.connection.driver_class">org.h2.Driver</property>
 9         <property name="hibernate.connection.url">jdbc:h2:mem:db1;DB_CLOSE_DELAY=-1;MVCC=TRUE</property>
10         <property name="hibernate.connection.username">sa</property>
11         <property name="hibernate.dialect">org.hibernate.dialect.H2Dialect</property>
12         <property name="hibernate.hbm2ddl.auto">update</property>
13         <property name="hibernate.show_sql">true</property>
14         <property name="hibernate.format_sql">true</property>
15         <property name="hibernate.generate_statistics">true</property>
16     </session-factory>
17 </hibernate-configuration>

 

(2)屬性文件方式配置Hibernate:

1 hibernate.connection.driver_class org.h2.Driver
2 hibernate.connection.url jdbc:h2:mem:db1;DB_CLOSE_DELAY=-1;MVCC=TRUE
3 hibernate.connection.username sa
4 hibernate.dialect org.hibernate.dialect.H2Dialect
5 hibernate.hbm2ddl.auto update
6 hibernate.show_sql true
7 hibernate.format_sql true
8 hibernate.generate_statistics true

2.實體映射:1對多、多對多

Hibernate的實體映射能夠採起XML和代碼註解兩種, .NET中的EntityFramework的實體映射也有對應的註解(特性)方式,但提供了讓實體類更加乾淨的代碼配置方式。不管是依賴注入仍是實體映射,Spring和Hibernate在這方面始終相對落後和繁瑣。

各類JAVA框架的核心歷來不是xml,框架的核心功能和核心對象纔是最重要的。註解配置的核心註解以下:

(1)@Entity:標註類爲實體。

(2)@Id和@GeneratedValue:前者標註POJO字段爲主鍵,後者標註字段爲數據庫自動生成。

(3)@OneToMany和@ManyToOne:在關聯字段上標註1對多和多對1。

(4)@ManyToMany:在關聯字段上標註多對多,cascade參數指定級聯處理規則。

(5)@Version:標註字段爲樂觀併發控制版本字段。

下面分別演示常見的1對多、多對多的映射配置。

(1)1對多:Category-Post

Category代碼:

 1 @Entity
 2 public class Category {
 3 
 4     @Id
 5     @GeneratedValue
 6     private int id;
 7 
 8     private String Name;
 9 
10     @OneToMany
11     private List<Post> posts = new ArrayList<Post>();
12 
13     public int getId() {
14         return id;
15     }
16 
17     public void setId(int id) {
18         this.id = id;
19     }
20 
21     public String getName() {
22         return Name;
23     }
24 
25     public void setName(String name) {
26         Name = name;
27     }
28 
29     public List<Post> getPosts() {
30         return posts;
31     }
32 
33     public void setPosts(List<Post> posts) {
34         this.posts = posts;
35     }
36 }

Post代碼:

@Entity
public class Post {
    @Id
    @GeneratedValue
    private int id;

    private String Name;

    private String Text;

    @ManyToOne
    private Category category;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return Name;
    }

    public void setName(String name) {
        Name = name;
    }

    public String getText() {
        return Text;
    }

    public void setText(String text) {
        Text = text;
    }

    public Category getCategory() {
        return category;
    }

    public void setCategory(Category category) {
        this.category = category;
    }
}

 

(2)多對多+樂觀鎖:User-Role

User代碼:

 1 @Entity
 2 public class User {
 3     @Id
 4     @GeneratedValue
 5     private int id;
 6 
 7     private String userName;
 8 
 9     private String password;
10 
11     @Version
12     private long version;
13 
14     @ManyToMany(cascade = CascadeType.ALL)
15     private List<Role> roles = new ArrayList<Role>();
16 
17     public int getId() {
18         return id;
19     }
20 
21     public void setId(int id) {
22         this.id = id;
23     }
24 
25     public String getUserName() {
26         return userName;
27     }
28 
29     public void setUserName(String userName) {
30         this.userName = userName;
31     }
32 
33     public String getPassword() {
34         return password;
35     }
36 
37     public void setPassword(String password) {
38         this.password = password;
39     }
40 
41     public long getVersion() {
42         return version;
43     }
44 
45     public void setVersion(long version) {
46         this.version = version;
47     }
48 
49     public List<Role> getRoles() {
50         return roles;
51     }
52 
53     public void setRoles(List<Role> roles) {
54         this.roles = roles;
55     }
56 
57 }

Role代碼:

 1 @Entity
 2 public class Role {
 3     @Id
 4     @GeneratedValue
 5     private int id;
 6 
 7     private String roleName;
 8 
 9     @ManyToMany(cascade = CascadeType.ALL)
10     private List<User> users = new ArrayList<User>();
11 
12     public int getId() {
13         return id;
14     }
15 
16     public void setId(int id) {
17         this.id = id;
18     }
19 
20     public String getRoleName() {
21         return roleName;
22     }
23 
24     public void setRoleName(String roleName) {
25         this.roleName = roleName;
26     }
27 
28     public List<User> getUsers() {
29         return users;
30     }
31 
32     public void setUsers(List<User> users) {
33         this.users = users;
34     }
35 }

 

3.會話工廠與會話:SessionFactory&Session

(1)會話上下文SessionFactory

SessionFactory始終是Hibernate的核心對象.經過Configuration建立的SessionFactory是Hibernate ORM的核心對象。Hibernate 4.3.5和Hibernate 5.x可使用一致的代碼建立SessionFactory,但5.x須要引入jta(javax.transaction),不然建立失敗。

 1     public SessionFactory sessionFactory() {
 2 
 3         org.hibernate.cfg.Configuration configuration = new org.hibernate.cfg.Configuration();
 4 
 5         configuration.addAnnotatedClass(User.class);
 6         configuration.addAnnotatedClass(Role.class);
 7         configuration.addAnnotatedClass(Category.class);
 8         configuration.addAnnotatedClass(Post.class);
 9 
10         SessionFactory sessionFactory = configuration.buildSessionFactory(new StandardServiceRegistryBuilder().build());
11         return sessionFactory;
12 
13     }

(2)會話Session

Session對象相似於EntityFramework中DbContext對象。Hibernate中經過SessionFactory獲取Session,有2種方式openSession()和 getCurrentSession()。openSession方式獲取單個打開的Session,須要本身寫代碼關閉。getCurrentSession方式則能夠獲取自動管理的Session對象,這是依賴CurrentSessionContext接口的實現類來支持的,能夠經過配置hibernate.current_session_context_class來適配,取值"jta","thread"和"managed"分別對應三個實現類。使用getCurrentSession時雖然不須要手動管理Session的關閉,可是須要手動管理Transaction事務的開啓和關閉。在Spring中繼承Hibernate時,Spring提供了CurrentSessionContext的實現類SpringJtaSessionContext,避免了咱們手動管理事務。在不使用Spring的Servlet環境中,咱們能夠選擇使用Filter+getSession方式使用Session。也能夠配置成"thread"+手動管理事務方式。

Filter+getSession能夠直接使用click-extras程序包中的Filter和SessionContext,最好是複用並修改其源碼,其中SessionContext的實現核心是使用類型爲ThreadLocal<Session>的靜態字段實現線程級別的Session共享。

click-extras的pom以下:

1 <dependency>
2     <groupId>org.apache.click</groupId>
3     <artifactId>click-extras</artifactId>
4     <version>2.3.0</version>
5 </dependency>

使用"thread"+手動管理事務方式須要先配置hibernate.properties屬性文件:hibernate.current_session_context_class thread。

1     private void Test(SessionFactory factory)
2     {
3         factory.getCurrentSession().beginTransaction();
4         Query query = factory.getCurrentSession().createSQLQuery("select * from User where userName=?").addEntity(User.class);
5         User user = (User) query.uniqueResult();
6         factory.getCurrentSession().getTransaction().commit();
7     }

4.查詢:SQL原生查詢、HQL通用查詢、Criteria條件查詢

(1)SQL原生查詢:

原生查詢使用Query接口的子接口SQLQuery。經過session能夠建立該接口的實例。下面的代碼中hsqldb的參數化查詢佔位符是"?"。爲了便於使用,使用了SQLQuery的addEntity方法配置查詢對應的實體類型。

1     private User SqlQuery() {
2         Session session = SessionContext.getSession();
3         Query query = session.createSQLQuery("select * from User where userName=?").addEntity(User.class);
4         query.setString(0, "admin");
5         return (User) query.uniqueResult();
6     }

(2)HQL通用查詢:

Hibernate使用Query接口,經過自定義的HQL實現通用查詢,HQL提供了一箇中間語言,屏蔽了不一樣數據庫的語法差別。經過session能夠建立Query接口的實例。

1     private User SqlQuery() {
2         Session session = SessionContext.getSession();
3         Query query = session.createQuery("from User where userName=:userName");
4         query.setString("userName", "admin");
5         return (User) query.uniqueResult();
6     }

(3)Criteria條件查詢:

Hibernate經過Criteria對象提供對自動化查詢的方法級別的支持,輔助類Restrictions提供了大量靜態方法建立Criteria對象,最大的做用就是防止寫錯SQL關鍵字。Java中沒有相似.NET中Linq同樣的語言集成查詢

1     private User CriteriaQuery() {
2         Session session = SessionContext.getSession();
3         Criteria query = session.createCriteria(User.class);
4         query.add(Restrictions.eq("userName", "admin"));
5         return (User) query.uniqueResult();
6     }

5.事務

 Transanction接口是Hibernate中封裝事務的接口,支持JDBC數據庫事務和JTA分佈式事務,能夠經過Session對象使用Transanction進行事務管理。JAT事務則與JTA容器緊密相關,之後再續。

1     private void Test(SessionFactory factory) {
2         factory.getCurrentSession().beginTransaction();
3         Query query = factory.getCurrentSession().createSQLQuery("select * from User where userName=?")
4                 .addEntity(User.class);
5         User user = (User) query.uniqueResult();
6         factory.getCurrentSession().getTransaction().commit();
7

參考

(1)http://docs.jboss.org/hibernate/orm/5.0/quickstart/html/

(2)https://www.ibm.com/developerworks/cn/java/j-lo-jta/

(3)https://www.ibm.com/developerworks/cn/java/j-lo-hibernate3/

相關文章
相關標籤/搜索