转载请注明: TheViper http://www.cnblogs.com/TheViper
SessionFactory接口
一个SessionFactory接口对应一个数据存储源。特点:
1.线程安全。即它的同一个实例可以被应用的多个线程共享。
2.重量级。创建和销毁它的实例所花费的代价很大,所以应用的一个数据库对应一个SessionFactory实例,在初始化时创建。
Session接口
持久化管理器,复杂和持久化的相关操作。特点:
1.不是线程安全。2.实例是轻量级的。
Hibernate的映射类型
有一点说明下,如果需要存储的字符串比较多,而且字符串长度也不好掌握,数据库就不好用varchar.这时可以像这样.
<property name="content"> <column name="content" sql-type="mediumtext" /> </property>
pojo那边的content类型设置成String.可以看到上面表里面sql字段类型没有mediumtext.这样做就可以让java String数据库直接映射成medirmtext了。我用的是mysql,不知道其他数据库是不是这样的。
映射主键
1.单个主键
<id name="feeling_id" type="java.lang.Integer"> <column name="feeling_id " length="20" /> <generator class="native" /> </id>
2.复合主键
(1)与其他表无关联
比如:名字(name),好友名字(friend_name),好友分类(sort),name和friend_name组成复合主键user_friend_pk。
对user_friend_pk,单独创建一个User_Friend_pk类,继承Serializable接口,并重写hashCode(),equals(Object obj)方法。
对pojo User_Friend类,对User_Friend_pk setter,getter即可。
映射文件:
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="cls.User_Friend" table="user_friends"> <composite-id name="user_friend_pk" class="cls.User_Friend_pk"> <key-property name="name" column="name" type="java.lang.String" length="20"></key-property> <key-property name="friend_name" column="friend_name" type="java.lang.String" length="20"></key-property> </composite-id> <property name="sort" type="java.lang.String"> <column name="sort" length="20" /> </property> </class> </hibernate-mapping>
(2)与其他表有关联
比如:User表(id,name) User_Friend表和上面一样,只不过把name和friend_name换成id和friend_id,这就形成了关联。
这和上面的区别在于映射文件。
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="model.cls.User_Friend" table="user_friends"> <composite-id name="user_friend_pk" class="model.cls.User_Friend_pk"> <key-many-to-one name="ids" class="model.cls.User"> <column name="id"></column> </key-many-to-one> <key-many-to-one name="friend_ids" class="model.cls.User"> <column name="friend_id"></column> </key-many-to-one> </composite-id> <property name="sort" type="java.lang.String"> <column name="sort" length="20" /> </property> </class> </hibernate-mapping>
下面来个复杂点的。
student表,班级(classId),学号(studentId).这两个可以确定一个学生,构成复合主键。
course表,课程id(courseId),课程名(courseName).
很显然,这两个是多对多的关系。我们用<many-to-many>构建单向的多对多映射。当然也可以用两个一对多关系来构建。
映射文件student.hbm.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="cls.Student" table="student"> <composite-id name="studentPk" class="cls.StudentPk"> <key-property name="studentId" column="studentId" type="java.lang.Integer"></key-property> <key-property name="classId" column="classId" type="java.lang.Integer"></key-property> </composite-id> </class> </hibernate-mapping>
course.hbm.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="cls.Course" table="course"> <id name="courseId" type="java.lang.Integer"> <column name="courseId " length="20" /> <generator class="native" /> </id> <property name="courseName" type="java.lang.String"> <column name="courseName" length="20" /> </property> <set name="StudentCourse" table=‘StudentCourse‘> <key column="courseId" /> <many-to-many class="cls.Student"> <column name="studentId"></column> <column name="classId"></column> </many-to-many> </set> </class> </hibernate-mapping>
关于<set>上的inverse属性
作用:决定是由谁来维护表和表之间的关系的。这里有个前提,这两个表之间必须是双向关联,道理很简单,比如一对多关系,在“一”那边设置了“一对多” inverse=true,让“多”那边来维护,但没有设置“多“那边的集合或类对象的映射,让”多“那边怎么去维护。
比如上面复合主键的第二个例子,我把sort(分类)变成一个friend_category_id,另外建个表friend_category。
映射文件Friend_category.hbm.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="cls.Friend_Category" table="friend_category"> <id name="friend_category_id" type="java.lang.Integer"> <column name="friend_category_id " length="20" /> <generator class="native" /> </id> <property name="category_name" type="java.lang.String"> <column name="category_name" length="20" /> </property> <set name="user_friends" inverse="true"> <key column="friend_category_id" /> <one-to-many class="cls.User_Friend" /> </set> </class> </hibernate-mapping>
User_friend.hbm.xml
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN" "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd"> <hibernate-mapping> <class name="cls.User_Friend" table="user_friends"> <composite-id name="user_friend_pk" class="cls.User_Friend_pk"> <key-property name="name" column="name" type="java.lang.String" length="20"></key-property> <key-property name="friend_name" column="friend_name" type="java.lang.String" length="20"></key-property> </composite-id> <!-- <many-to-one name="friend_categorys" column="friend_category_id" --> <!-- class="cls.Friend_Category"> --> <!-- </many-to-one> --> <!-- <property name="sort" type="java.lang.String"> --> <!-- <column name="sort" length="20" /> --> <!-- </property> --> </class> </hibernate-mapping>
为两表插入数据
Friend_Category fc = new Friend_Category(); fc.setCategory_name("好基友"); User_Friend_pk pk = new User_Friend_pk(); pk.setFriend_name("daut"); pk.setName("TheViper"); User_Friend uf = new User_Friend(); uf.setUser_friend_pk(pk); uf.setFriend_categorys(fc); // uf.setSort("好基友"); session.save(fc); session.save(uf);
Hibernate: insert into friend_category (category_name) values (?) Hibernate: insert into user_friends (name, friend_name) values (?, ?)
可以看到在user_friends表中并没有插入friend_category_id字段。
还有一点,既然是双向关联了,上面插入数据的代码我也完全可以这样写。
Friend_Category fc = new Friend_Category(); fc.setCategory_name("好基友"); User_Friend_pk pk = new User_Friend_pk(); pk.setFriend_name("daut"); pk.setName("TheViper"); User_Friend uf = new User_Friend(); uf.setUser_friend_pk(pk); Set user_friends = new HashSet(); user_friends.add(uf); fc.setUser_friends(user_friends); session.save(fc); session.save(uf);
Hibernate: insert into friend_category (category_name) values (?) Hibernate: insert into user_friends (friend_category_id, name, friend_name) values (?, ?, ?)
可以看到,这次user_friends表总算插入了friend_category_id字段,但是一看数据,friend_category_id字段居然木有数据。
因为你已经inverse=true了,两表关系就由user_friends(User_Friend类)来维护了,但是上面的代码却是通过Friend_Category的setUser_friends()添加关联。这时应该是User_Friend的setFriend_categorys()才对。