hibernate查漏补缺1

转载请注明: 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()才对。

 

posted on 2014-11-21 18:04  TheViper_  阅读(355)  评论(0编辑  收藏  举报