0

理论上, session.get() 方法应该总是命中数据库,无论实体是否存储在缓存中。但是每当我使用 session.get() 或 session.load() 时,两者都不会第二次访问数据库。

    Session session = factory.openSession();
    tx = session.beginTransaction();
    Customer cust = (Customer)session.get(Customer.class,2);    
    System.out.println(cust.getCid()+","+cust.getFirstName()+","+cust.getLastName()+","+cust.getPhone());                                
    Customer cust2 = (Customer)session.get(Customer.class,2);          
    System.out.println(cust2.getCid()+","+cust2.getFirstName()+","+cust2.getLastName()+","+cust2.getPhone());
    tx.commit();
    session.close();

这是输出,

Hibernate: select customer0_.cid as cid1_1_0_, customer0_.firstName as firstNam2_1_0_, customer0_.lastName as lastName3_1_0_, customer0_.email as email4_1_0_, customer0_.phone as phone5_1_0_, customer0_.aid as aid6_1_0_ from mycustomers customer0_ where customer0_.cid=?
2,Sam,pp,9799999999
2,Sam,pp,9799999999

选择查询只执行一次,下次从缓存中检索。如果我也使用 session.load() 方法,则输出相同。

我在这里错过了什么吗?请说清楚。

4

1 回答 1

-1

这是这里发生的事情:

  1. 控制台上的第一个查询

    它总是会返回一个“代理”。例如,如果您这样做session.load(Customer.class, 2),它将返回一个代理对象。代理对象只有一个标识符值,没有别的。你可以想象它有点像这样。

        customer.id = 2;
        customer.fname = null;
        customer.lname = null;
        customer.address = null;
        //rest all properties are null
    

    每当您访问属性时,它都会访问数据库。在您的情况下,您会立即调用ust.getCid(),因此它将立即访问数据库以获取这些查询。因此,您在控制台中看到的第一个查询将出现在这两种情况下(即session.get()session.load()

    尝试这样做,看看你的控制台是什么样子的:

        Session session = factory.openSession();
        tx = session.beginTransaction();
        Customer cust = (Customer)session.get(Customer.class,2);    
        //do not call any getter.
    

    您将在控制台上看到差异。

     

  2. 为什么第二个查询没有出现

    Hibernate 二级缓存
    您正试图访问您之前访问过的同一个对象。然后 Hibernate 将(而不是再次从数据库中获取它)从二级缓存中获取它。

    您将在此页面上找到此方案的详细示例:Hibernate Second Level Cache。(只看最后一个例子,它与你得到的相似)

于 2017-03-03T05:57:20.157 回答