@Basic(fetch = FetchType.LAZY)不起作用?

时间:2010-01-21 20:13:20

标签: java hibernate jpa orm lazy-loading

我在Spring中使用JPA(Hibernate)。 当我想延迟加载Stirng属性时,我使用以下语法:

@Lob
@Basic(fetch = FetchType.LAZY)
public String getHtmlSummary() {
    return htmlSummary;
}

但是当我看到hibernate创建的sql时,似乎这个属性不是延迟加载的?我也在ANT脚本中使用这个类org.hibernate.tool.instrument.javassist.InstrumentTask来检测这个属性,但它似乎不起作用。

请帮帮我。

在Khosro。

7 个答案:

答案 0 :(得分:5)

延迟Lob加载需要字节码检测才能正常工作,所以默认情况下它在我所知道的任何JPA实现中都不可用。

最好的办法是将Lob放入一个单独的实体,如HtmlSummary,并使用一个懒惰加载的一对一关联。

答案 1 :(得分:3)

将FieldHandled与@Basic(fetch=FetchType.LAZY)一起使用:

public class myFile implements Serializable, FieldHandled
{

    private FieldHandler      fieldHandler;

    @Lob
    @Basic(fetch = FetchType.LAZY)
    @Column(name = "CONTENT")
    protected byte[]          content;

答案 2 :(得分:3)

@Entity
public class User implements FieldHandled {

    @Id
    private String uid;

    private String uname;

    private int age;

    @Lob
    @Basic(fetch = FetchType.LAZY)
    private byte[] img;

    private FieldHandler fieldHandler;

    public User() {
    }

    // getter() and setter() of uid, uname, age

    public byte[] getImg() {
        // if User user = new User() then fieldHandler is null
        // if User user = entityManager.find(User.class, "001") then fieldHandler is not null
       if(img != null) { 
           return img;
       }

       if (fieldHandler != null) { 
           return (byte[]) fieldHandler.readObject(this, "img", img);
       } else {
           return null;
       }  
    }

    public void setImg(byte[] img) {
        this.img = img;
    }

    public void setFieldHandler(FieldHandler fieldHandler) {
        this.fieldHandler = fieldHandler;
    }

    public FieldHandler getFieldHandler() {
        return fieldHandler;
    }
}

我使用Hibernate4 h2database。我确定延迟加载可以通过我的代码正常工作。

Hibernate:select user0_.uid as uid1_0_0_, user0_.age as age2_0_0_, user0_.uname as uname4_0_0_ from User user0_ where user0_.uid=?

Hibernate:select user_.img as img3_0_ from User user_ where user_.uid=?

如果使用repository.save(User)添加新用户即可,但更新用户将抛出异常

java.lang.ClassCastException: org.hibernate.bytecode.instrumentation.spi.LazyPropertyInitializer$1 cannot be cast to java.sql.Blob

我建议在repository.delete(userid)之前在一个交易中使用repository.save,然后它会正常工作。

答案 3 :(得分:3)

首先,您应该知道JPA规范明确规定LAZY只是对JPA提供商的暗示,因此它不是强制性要求。

要使基本类型的延迟抓取工作,您需要enable bytecode enhancement并明确将enableLazyInitialization配置属性设置为true

<plugin>
    <groupId>org.hibernate.orm.tooling</groupId>
    <artifactId>hibernate-enhance-maven-plugin</artifactId>
    <version>${hibernate.version}</version>
    <executions>
        <execution>
            <configuration>
                <enableLazyInitialization>true</enableLazyInitialization>
            </configuration>
            <goals>
                <goal>enhance</goal>
            </goals>
        </execution>
    </executions>
</plugin>

答案 4 :(得分:1)

从JPA的规范中他们说即使你使用注释属性来懒散地获取,也不能保证应用,因此属性可能会或可能不会被懒惰地加载(取决于JPA的实现者),但是如果您指定应该急切地获取它们,那么JPA实现者必须急切地加载它们。

底线:@Basic(fetch = FetchType.LAZY)可能有效,也可能无效,取决于JPA实施者。

答案 5 :(得分:0)

我认为它与EclipseLink类似,你需要启用编织,否则fetch设置不起作用。 编织需要字节码访问。 这有助于:https://stackoverflow.com/a/18423704/7159396

答案 6 :(得分:-3)

延迟提取仅适用于对其他实体或实体集合的引用。它不适用于String或int等值。