zoukankan      html  css  js  c++  java
  • Hibernate延迟加载问题

    Hibernate的映射配置单如下:

    <?xml version="1.0" encoding="GBK"?>
    <!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
    <hibernate-mapping>
        <class name="ext.eclipse.ide.auth.role.impl.UserImpl" table="Users">
            <id name="id">
                <column name="UID" length="15" />
            </id>
            <property name="name">
                <column name="NAME" length="20" />
            </property>
            <property name="password">
                <column name="PASSWORD" length="64" />
            </property>
            <set name="roles" table="User_Role" cascade="all-delete-orphan" fetch="join">
                <key column="USER_ID"></key>
                <many-to-many class="ext.eclipse.auth.role.impl.RoleImpl"
                    column="ROLE_ID"></many-to-many>
            </set>
        </class>
    </hibernate-mapping>

    可以看到,UserImpl和RoleImpl是many-to-many的关系,关系Set维护在UserImpl的roles属性中。

    set标签没有写lazy="false"属性,即是懒加载被设定为默认的true,这表示:

    当同一个事务对UserImpl进行查询的时候,如果roles没有被使用到,是不会对User_Role关系表查询的。

    很明显,这是为了提高效率。

    我对Session进行了封装:

        public List<?> query(String hql) {
            Session session = getSessionFactory().getCurrentSession();
    
            session.beginTransaction();
            Query query = null;
            try {
                query = session.createQuery(hql);
            } catch (Exception e) {
                e.printStackTrace();
            }
            List<?> result = query == null ? new ArrayList<Object>() : query.list();
            session.getTransaction().commit();
    
            return result;
        }

    按照以上的代码,如果我调用

    query("from UserImpl");

    获取UserImpl对象,当调用到user.getRoles()的时候,是会抛出懒加载异常的(Session不存在或者已经关闭)

    解决办法有四:

    1、提供Session出来,单独写一个包含getRoles的查询

    2、使用Hibernate.initialize(Object proxy)方法,强制加载

    3、在配置文件里设置lazy="false"(明显不可取)

    4、为配置文件增加fetch="join",然后hql改为“from UserImpl ui left join fetch ui.roles”

    推荐4方案。

  • 相关阅读:
    centos 给终端设快捷键
    centos 给鼠标右击添加 “打开终端” 菜单项
    centos 6.X 安装输入法
    centos U盘安装
    js的构造函数
    onresize的定义方式
    两个时间对比
    AMD和CMD的区别
    spring加载配置文件
    cglib代理
  • 原文地址:https://www.cnblogs.com/anrainie/p/2540326.html
Copyright © 2011-2022 走看看