-->
These old forums are deprecated now and set to read-only. We are waiting for you on our new forums!
More modern, Discourse-based and with GitHub/Google/Twitter authentication built-in.

All times are UTC - 5 hours [ DST ]



Forum locked This topic is locked, you cannot edit posts or make further replies.  [ 6 posts ] 
Author Message
 Post subject: n+1 problem
PostPosted: Mon Mar 26, 2007 4:34 am 
Regular
Regular

Joined: Thu Dec 22, 2005 7:47 am
Posts: 62
Location: Czech Republic
Hello,

i have encountered a strange n+1 problem and severe performance degradation on MSSQL with either official and/or opensource (jtds) drivers alike.

i believe i have the right mapping, and i suspect that maybe inheritence causes the problem (nothing else left :-).

Hibernate version: 3.2.2

Mapping documents: (just fragments)

There is many-to-one relation mapped as shown bellow -- default (lazy) mapping on both sides.

There are times when i must access plenty of that objects (cca 5000) and attributes from both of them are needed. In that case i use the Criteria and/or HQL query to explicitly fetch the objects (Criteria shown below).

from BookEntryOperation to BookEntry

Code:
    <!-- BookEntryOperation -->
    <subclass name="BookEntryOperation" discriminator-value="book_entry">

      <property name="newDate">
   <column name="new_date" not-null="false"/>
      </property>

      <property name="newAmount">
   <column name="new_amount" not-null="false"/>
      </property>

      <!-- should be uniq, 1:1, some db (MSSQL) have problems -->
      <many-to-one name="bookEntry" class="BookEntry" column="book_entry_id"
   not-null="false" unique="false"/>

    </subclass>


from BookEntry to BookEntryOperation

Code:
    <one-to-one name="operation" class="BookEntryOperation" cascade="all"
      property-ref="bookEntry"/>


Code between sessionFactory.openSession() and session.close():

When executing this criteria, i got the right sql query with left join (as shown below), BUT i also get n queries fetching the single objects (no matter what db engine -- MySQL, HSQLdb, MSSQL).

Code:
   Criteria c = session.createCriteria(BookEntryOperation.class);
   c.setFetchMode("bookEntry", FetchMode.JOIN)
       .createCriteria("bookEntry");
   Disjunction or = Restrictions.disjunction();
   or.add(Restrictions.between("date", getFromDate(), getToDate()));
   or.add(Restrictions.between("newDate",  getFromDate(), getToDate()));
   c.add(or);
   addOperationAssocCriteria(c);
   return c;


In case of MySQL or HSQLdb the speed is sufficient and the queries just run through my log returning almost immediatelly and i would say there is no performace loss (still strange to me, however, that though i got the right left join query and the objects shoul already be selected, the db runs another n queries to get them).

On MSSQL the performance is terrible. On MySQl or HSQL i do the queries (the 1 with left join and then the n to get that objects -- though the running of that n queries is strange to me as mentioned), do the computation, and generate the result (business object that gets later translated to HTML) in seconds. MSSQL runs the same queries and the generations tooks about 5 minutes -- during those 5 minute the n queries is written to log .

I did check for general db settings (as far as my knowledge goes) -- the autoCommit, transactionIsolation and everything seems to be just fine.

Full stack trace of any exception that occurs:

no exception

Name and version of the database you are using:

MySQL 5.0, HSQLdb 1.8, MSSQL 2000/2005

The generated SQL (show_sql=true):

Code:
9349 [main] DEBUG (SQL) org.hibernate.jdbc.AbstractBatcher.log(AbstractBatcher.java:393) -
    /* criteria query */ select
        this_.operation_id as operation1_10_1_,
        this_.typex as typex10_1_,
        this_.amount as amount10_1_,
        this_.datex as datex10_1_,
        this_.status as status10_1_,
        this_.description as descript7_10_1_,
        this_.confirmationNumber as confirma8_10_1_,
        this_.currency_id as currency9_10_1_,
        this_.account_id as account10_10_1_,
        this_.counter_party_id as counter11_10_1_,
        this_.organization_unit_id as organiz12_10_1_,
        this_.new_date as new16_10_1_,
        this_.new_amount as new17_10_1_,
        this_.book_entry_id as book18_10_1_,
        bookentry1_.book_entry_id as book1_6_0_,
        bookentry1_.ctime as ctime6_0_,
        bookentry1_.mtime as mtime6_0_,
        bookentry1_.creator as creator6_0_,
        bookentry1_.owner as owner6_0_,
        bookentry1_.business_id as business6_6_0_,
        bookentry1_.business_var_id as business7_6_0_,
        bookentry1_.purchase_date as purchase8_6_0_,
        bookentry1_.posting_date as posting9_6_0_,
        bookentry1_.settlement_date as settlement10_6_0_,
        bookentry1_.description as descrip11_6_0_,
        bookentry1_.import_warning as import12_6_0_,
        bookentry1_.user_override as user13_6_0_,
        bookentry1_.book_entry_type_id as book14_6_0_,
        bookentry1_.group_id as group15_6_0_,
        bookentry1_.import_file_id as import16_6_0_
    from
        operation this_
    inner join
        book_entry bookentry1_
            on this_.book_entry_id=bookentry1_.book_entry_id
    where
        this_.subclass='book_entry'
        and (
            this_.datex between ? and ?
            or this_.new_date between ? and ?
        )
9481 [main] DEBUG (SQL) org.hibernate.jdbc.AbstractBatcher.log(AbstractBatcher.java:393) -
    /* load cz.facility.facicash.app.core.BookEntryOperation */ select
        bookentryo0_.operation_id as operation1_10_0_,
        bookentryo0_.typex as typex10_0_,
        bookentryo0_.amount as amount10_0_,
        bookentryo0_.datex as datex10_0_,
        bookentryo0_.status as status10_0_,
        bookentryo0_.description as descript7_10_0_,
        bookentryo0_.confirmationNumber as confirma8_10_0_,
        bookentryo0_.currency_id as currency9_10_0_,
        bookentryo0_.account_id as account10_10_0_,
        bookentryo0_.counter_party_id as counter11_10_0_,
        bookentryo0_.organization_unit_id as organiz12_10_0_,
        bookentryo0_.new_date as new16_10_0_,
        bookentryo0_.new_amount as new17_10_0_,
        bookentryo0_.book_entry_id as book18_10_0_
    from
        operation bookentryo0_
    where
        bookentryo0_.book_entry_id=?
        and bookentryo0_.subclass='book_entry'


Debug level Hibernate log excerpt:

see above

- - - -

I suspect that maybe the inheritance mapping has something to do with it (the last idea i have) -- in this particular case i use table-per-class-hierarchy strategy and although i select only _BookEntryOperation_ in my query and there are none superclasses defined, the implementation asks with a single query to fetch all the attributes of a given class in case it would be a super class? If so, is there any work arround?

Thanks a lot for any comment, martin.


Top
 Profile  
 
 Post subject: Try HQL
PostPosted: Wed Mar 28, 2007 1:21 am 
Newbie

Joined: Wed Mar 28, 2007 1:05 am
Posts: 8
Hi ,
I think problem doesn't lie in the inheritance the extra queries that its firing is because of Criteria, try using HQL for this.

cheers,


Top
 Profile  
 
 Post subject: maybe sql query
PostPosted: Wed Mar 28, 2007 9:13 am 
Regular
Regular

Joined: Thu Dec 22, 2005 7:47 am
Posts: 62
Location: Czech Republic
Sorry, but that was not helpfull, i tried already (without knowing the internals of hibernate i would assume that the part that parses the criteria and hql will result in the same actions).

I was thinking about sql (session.createSQLQuery) but the associations and composites in my model makes the interface hard to use and it will be better just to do in plain sql.


Top
 Profile  
 
 Post subject:
PostPosted: Wed Mar 28, 2007 10:45 am 
Expert
Expert

Joined: Tue Jul 11, 2006 10:21 am
Posts: 457
Location: Columbus, Ohio
Would this work for ya?

Code:
   Criteria c = session.createCriteria(BookEntryOperation.class);
       .createCriteria("bookEntry", Criteria.LEFT_JOIN);
   Disjunction or = Restrictions.disjunction();
   or.add(Restrictions.between("date", getFromDate(), getToDate()));
   or.add(Restrictions.between("newDate",  getFromDate(), getToDate()));
   c.add(or);
   addOperationAssocCriteria(c);
   return c;


Top
 Profile  
 
 Post subject:
PostPosted: Wed Mar 28, 2007 5:04 pm 
Regular
Regular

Joined: Thu Dec 22, 2005 7:47 am
Posts: 62
Location: Czech Republic
again, sorry to say that, that was not the solution -- just another way how to achieve the same result.

In fact, the code:

Code:
   Criteria c = session.createCriteria(BookEntryOperation.class);
   Criteria c2 = c.createCriteria("bookEntry", Criteria.LEFT_JOIN);
   Disjunction or = Restrictions.disjunction();
   or.add(Restrictions.between("date", getFromDate(), getToDate()));
   or.add(Restrictions.between("newDate",  getFromDate(), getToDate()));
   c.add(or);
   addOperationAssocCriteria(c);
   return c;


is the same as i had before -- i just used the version with FetchMode.

The reason i suspect inheritance is, that the 1st query to database is just ok -- it is a proper join of the two tables. But after that, the hibernate should create objects of a proper class, each row may contain a different (sub)class and that is why for each row there has to be a separate select -- to choose all the attributes of a subclass.

I can live with that, the HQLdb and MySQL databases are just fine -- but the speed on MSSQL is terrific.


Top
 Profile  
 
 Post subject: probable solution
PostPosted: Mon Apr 09, 2007 12:29 pm 
Regular
Regular

Joined: Thu Dec 22, 2005 7:47 am
Posts: 62
Location: Czech Republic
hi, just found myself a solution :-).

The relationship is parent->child (BookEntry -> BookEntryOperation). if i use the outer join fetch from parent (BookEntry), it is ok, if i use it from the child (BookEntryOperation) subsequent selects are issued.

Schematically:

Code:
select be from BookEntry be
left join fetch be.operation beop


is fine (only one select is issued -- and i use java to obtain the proper children), while:

Code:
select beop from BookEntryOperation beop
left join fetch beop.bookEntry be



is wrong (n+1 selects are issued).

Dont know if it is a bug or feature?


Top
 Profile  
 
Display posts from previous:  Sort by  
Forum locked This topic is locked, you cannot edit posts or make further replies.  [ 6 posts ] 

All times are UTC - 5 hours [ DST ]


You cannot post new topics in this forum
You cannot reply to topics in this forum
You cannot edit your posts in this forum
You cannot delete your posts in this forum

Search for:
© Copyright 2014, Red Hat Inc. All rights reserved. JBoss and Hibernate are registered trademarks and servicemarks of Red Hat, Inc.