從以下5個方面學習hibernate ORM。
(1)配置檔案:hibernate.cfg.xml XML檔案和hibernate.properties屬性檔案
(2)實體對映:1對多、多對多
(3)會話工廠與會話:SessionFactory&Session
(4)查詢:SQL原生查詢、HQL通用查詢、Criteria條件查詢
(5)事務:Transanction
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:
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/