Гибернация: двунаправленное ограничение внешнего ключа «один ко многим» не выполняется - PullRequest
2 голосов
/ 27 ноября 2011

Я хочу реализовать отношение один-ко-многим, используя Hibernate (на Java).У меня есть две сущности:

  • Эксперимент
  • ExperimentGroup

Experiment имеет много ExperimentGroups.Я пытался настроить это отношение «один ко многим», как рекомендует официальный документ Hibernate: http://docs.jboss.org/hibernate/core/3.6/reference/en-US/html/example-parentchild.html#example-parentchild-cascades

Мой файл конфигурации:

<class name="ExperimentImpl" table="Experiment">
    <id name="id" type="int" column="id">
        <generator class="increment" />
    </id>
    <!-- One to Many   -->
    <set name="experimentGroups" table="ExperimentGroup" 
                lazy="false" fetch="select" cascade="all" inverse="true">
            <key>
                <column name="Experiment_id" not-null="true" />
            </key>
            <one-to-many class="ExperimentGroupImpl" />
    </set>  
</class>
<class name="ExperimentGroupImpl" table="ExperimentGroup">
    <id name="id" type="int" column="id">
        <generator class="increment" />
    </id>
    <many-to-one name="experiment" class="ExperimentImpl" fetch="select">
            <column name="Experiment_id" not-null="true" />
    </many-to-one>
</class>

Добавление нового ExperimentGroup вExperiment работает нормально, но удаление Эксперимента (и всех его Групп Эксперимента) вызывает исключение:

Невозможно удалить или обновить родительскую строку: сбой ограничения внешнего ключа (testdb. * 1023)*, CONSTRAINT FK9C71D86238B5500C ИНОСТРАННЫЙ КЛЮЧ (Experiment_id) ССЫЛКИ Experiment (id))

Мой код ExperimentDAO выглядит следующим образом:

public void deleteExperiment(Experiment experiment) 
         throws DAOException
    {
        Session session = null;
        Transaction t = null;
        try{
            session = HibernateUtil.getSessionFactory().openSession();
            t = session.beginTransaction();
                session.delete(experiment);
            t.commit();
            session.flush();
            session.close();
        }catch (HibernateException e)
        {
            e.printStackTrace();
            if (t != null)
                t.rollback();

            if (session != null)
                session.close();

            throw new DAOException(e, "Could not delete Experiment");
        }
    }

РЕДАКТИРОВАТЬ: Код DAO для создания Experiment и ExperimentGroup:

public Experiment createExperiment(String name, String description, RegisteredUser owner)
{
    Transaction tx = null;
    Session session = null;
    try
    {
        session = HibernateUtil.getSessionFactory().openSession();

        tx= session.beginTransaction();
        ExperimentImpl e = new ExperimentImpl();

        e.setName(name);
        e.setDescription(description);
        e.setOwner(owner);

        owner.getOwnedExperiments().add(e);

        session.save(e);
        tx.commit();
        session.flush();
        session.close();

        return e;
    } catch (HibernateException ex )
    {
        if (tx != null)
            tx.rollback();

        if (session!=null)
            session.close();

        throw ex;
    }
}




public ExperimentGroup createAndAddExperimentGroup(Experiment experiment, String experimentGroupName, Date startDate, Date endDate) throws ArgumentException
    {

        Transaction t = null;
        Session session = null;

        try
        {
            session = HibernateUtil.getSessionFactory().openSession();
            t = session.beginTransaction();
                ExperimentGroupImpl g = new ExperimentGroupImpl();
                g.setEndDate(endDate);
                g.setStartDate(startDate);
                g.setName(experimentGroupName);
                g.setExperiment(experiment);

                experiment.getExperimentGroups().add(g);
                g.setExperiment(experiment);

                session.save(g);
                session.update(experiment);
            t.commit();
            session.flush();
            session.close();

            return g;

        }catch(HibernateException e)
        {
            if (t!=null)
                t.rollback();

            if (session!= null)
                session.close();

            throw e;
        }
    }

Существует множество других свойств для Experiment, таких как имя, описание и т. Д., Которые я удалил из файла конфигурациипотому что они не влияют на отношение один-ко-многим.

Есть предложения?Что я не прав?

1 Ответ

1 голос
/ 27 ноября 2011

У меня сложилось впечатление, что вы пытаетесь удалить отдельную сущность, которую вы создали вручную.Я бы попытался использовать первичный ключ эксперимента, который вы хотите удалить, и загрузить перед тем, как его удалить:

session = HibernateUtil.getSessionFactory().openSession();
t = session.beginTransaction();
ExperimentImpl persistentExperiment = session.load(ExperimentImpl.class, experiment.getId());
session.delete(persistentExperiment);
t.commit();
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...