当前位置:首页 » PHP技术

如何解决 hibernate 自动更新实体类的问题

2018-06-13 08:21 本站整理 浏览(2)

如何解决 hibernate 自动更新实体类的问题

示例:

GrabOrder grabOrder = (GrabOrder)this.grabOrderDao.getCurrentSession().get(GrabOrder.class,grabOrderId);
grabOrder.setNickname(TimeHWUtil.getCurrentDateTime());

 

上述代码 会执行两条SQL:

查询和更新

为什么会更新呢?我明明没有调用DAO的update方法啊!!!

但是这里我调用了实体类的setter方法,

hibernate有一个机制:Hibernate dirty-checks,会同步session缓存和数据库的数据.

如何解决 自动更新的问题:

方案一:把实体类由持久态变为脱管态(游离态)

this.grabOrderDao.getCurrentSession().evict(grabOrder);

 缺点:懒加载的对象无法加载,因为此时session相当于close了

而且会报错:org.hibernate.LazyInitializationException: could not initialize proxy - no Session

报错的代码:

this.grabOrderDao.getCurrentSession().clear();
        System.out.println(grabOrder.getHouseInfo().getHouseOwner().getId());

 

 

更好的操作步骤:

(1)先把要访问的属性加载出来,可以通过提前使用,或者Hibernate.initialize(dept)

(2)再调用evict方法或clear方法

什么叫提前使用呢?即提前调用getter方法

正确:

System.out.println(grabOrder.getHouseInfo().getHouseOwner().getId());
        this.grabOrderDao.getCurrentSession().clear();
        System.out.println(grabOrder.getHouseInfo().getHouseOwner().getId());

 

 

方案二:把实体类变为只读

this.grabOrderDao.setReadOnly(grabOrder, readonly);

 但是这招不是万灵药,有例外,看官网:

12.2.2.2. Unidirectional one-to-many and many-to-many
Hibernate treats unidirectional one-to-many and many-to-many associations owned by a read-only entity the same as when owned by an entity that is not read-only.

修改 单向的一对多和多对多的成员变量时,依然会触发hibernate的自动更新.

有什么万全之策呢?

(a)采用读写分离,把操作放在只读的数据库连接中

 

(b)采用只读事务

当方法名是updateTestLazy 时会自动更新

但是我修改方法名为getTestLazy时,就不会自动更新了

为什么呢?看看spring 事务配置


 

只读的对象 不保险,所以我们使用只读的事务或只读的数据库连接.

 

如何解决 json序列化时,自动加载lazy对象导致对象多层嵌套的问题?

使用 SimpleBeanPropertyFilter过滤不需要的字段

 /***
     *
     * @param xpath : 参考 com/kunlunsoft/util/objectxpath/ObjectXpath.java
     * @return
     */
    public String toJson(String column, String... xpath) {
        if (this.result == null) {//modified by huangweii @2015-08-17
            System.out.println("this.result is null");
        } else {
            if (this.result != false) {
//                this.errorMessage = null;
                this.setErrorMessage(null);
            }

        }
        if (ValueWidget.isNullOrEmpty(xpath)
                || ValueWidget.isNullOrEmpty(xpath[0])) {
            return HWJacksonUtils.getJsonP(this);
        }

//        final ObjectXpath oxp = new ObjectXpath();
        int length = xpath.length;
//        final Object[] filterObjects = new Object[length];
        List filterObjList = new ArrayList();
        for (int i = 0; i < length; i++) {
            Object o = XPathParser.process(this.getValue(), xpath[i]);
            if (o instanceof List) {
                filterObjList.addAll((List) o);
            } else {
                filterObjList.add(o);
            }
//            filterObjects[i] = o;
        }
        final Object[] filterObjects = filterObjList.toArray();
        return HWJacksonUtils.getJsonP(this, new DefaultJsonFilterDecide() {
            @Override
            public boolean isFilteColumn(Object pojo, Object propertyVal, String column2) {
                if (ValueWidget.isNullOrEmpty(column)) {
                    if (SystemHWUtil.isEquals(propertyVal, filterObjects)) {
                        return true;
                    }
                } else {
                    if (SystemHWUtil.isEquals(pojo, filterObjects) && (column2.equals(column))) {
                        return true;
                    }
                }
                return false;
            }
        });
    }