hibernate каскадное удаление, почему бы не один удалить на внешний ключ? - PullRequest
2 голосов
/ 24 августа 2010

Мне интересно, почему hibernate генерирует 1 удаление для каждой сущности в дочерней таблице вместо использования одного удаления для внешнего ключа

Вот файл hibernate.cfg.xml (Нет, я не следующая SO: -t

<hibernate-configuration>
    <session-factory>
        <property name="hibernate.connection.url">jdbc:hsqldb:file:testdb;shutdown=true</property>
        <property name="hibernate.connection.driver_class">org.hsqldb.jdbcDriver</property>
        <property name="hibernate.connection.username">sa</property>
        <property name="hibernate.connection.password"></property>
        <property name="hibernate.connection.pool_size">0</property>
        <property name="hibernate.dialect">org.hibernate.dialect.HSQLDialect</property>
        <property name="hibernate.show_sql">true</property>
        <property name="hibernate.format_sql">true</property>
        <property name="hbm2ddl.auto">auto</property>
        <mapping file="entities/Question.hbm.xml"/>
        <mapping file="entities/Answer.hbm.xml"/>

    </session-factory>

Question.hbm.xml

<hibernate-mapping>
   <class name="entities.Question">
      <id name="id">
         <generator class="native" />
      </id>
      <property name="title" not-null="true">
      </property>

      <property name="question" type="text" not-null="true">
      </property>

      <bag name="answers" inverse="true" cascade="all,delete-orphan" >
         <key>
            <column name="questionId" index="answer_questionId_idx" not-null="true"/>
         </key>
         <one-to-many class="entities.Answer"  />
      </bag>

      <property name="created" update="false" >
         <column name="created" not-null="true" index="answer_created_idx"></column>
      </property>
      <property name="lastUpdated">
         <column name="lastUpdated"  not-null="true" index="answer_lastUpdated_idx"></column>
      </property>
   </class>
</hibernate-mapping>

Answer.hbm.xml

<hibernate-mapping>
   <class name="entities.Answer">
      <id name="id">
         <generator class="native" />
      </id>

      <property name="answer" type="text" not-null="true">
      </property>

      <property name="created" update="false" >
            <column not-null="true" name="created" index="question_created_idx"></column>
      </property>

      <property name="lastUpdated" >
            <column name="lastUpdated" not-null="true" index="question_lastUpdated_idx"></column>
                  </property>

      <many-to-one  name="question" column="questionId" not-null="true" update="false">
      </many-to-one>
   </class>
</hibernate-mapping>

В моей базе данных есть 1 вопрос и 2 ответа, этот тестовый код:

Session session = factory.openSession();
Transaction t = session.beginTransaction();
Question q = (Question) session.load(Question.class,1);
session.delete(q);
t.commit();
session.close();

Я бы ожидал, что он сгенерирует SQL, например,

select .... from Questions where id = 1;
delete from Answers where questionId=1;
delete from Question where id=1;

Т.е. просто выполните одно удаление, чтобы выполнить каскадное удаление ответов, вместо этого он загружает все ответы и выдаетодно удаление за ответ, например:

select
    question0_.id as id0_0_,
    question0_.title as title0_0_,
    question0_.question as question0_0_,
    question0_.created as created0_0_,
    question0_.lastUpdated as lastUpda5_0_0_ 
from
    Question question0_ 
where
    question0_.id=?

select
    answers0_.questionId as questionId0_1_,
    answers0_.id as id1_,
    answers0_.id as id1_0_,
    answers0_.answer as answer1_0_,
    answers0_.created as created1_0_,
    answers0_.lastUpdated as lastUpda4_1_0_,
    answers0_.questionId as questionId1_0_ 
from
    Answer answers0_ 
where
    answers0_.questionId=?

delete   from   Answer  where     id=?
delete   from   Answer  where     id=?
delete   from   Question where     id=?

Почему, и могу ли я с этим что-нибудь сделать?

Редактировать, в ответ на Нейта Заугга, я могу заставить БД сделатькаскадное удаление, установив on-delete = "cascade" в сопоставлении клавиш один-ко-многим, меня больше интересует, почему hibernate делает то, что делает, а не удаляет один в таблице ответов, и еще что-то не так с моимотображения.

1 Ответ

1 голос
/ 24 августа 2010

Разве вы не можете настроить DMBS для каскадного удаления отношений?Это действительно легко сделать.

Редактировать: Попробуйте это <one-to-many class="entities.Answer" lazy="false" cascade="all" />

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...