EntityManager.merge ничего не делает - PullRequest
12 голосов
/ 31 января 2012

У меня есть объект User:

@Entity
@Table( name = "bi_user" )
@SequenceGenerator( name = "USER_SEQ_GEN", sequenceName = "USER_SEQUENCE" )
public class User
        extends DataObjectAbstract<Long>
{
    private static final long serialVersionUID = -7870157016168718980L;

    /**
     * key for this instance. Should be managed by JPA provider.
     */
    @Id
    @GeneratedValue( strategy = GenerationType.SEQUENCE, generator = "USER_SEQ_GEN" )
    private Long key;

    /**
     * Username the user will use to login.  This should be an email address
     */
    @Column( nullable=false, unique=true)
    private String username;

    // etc. other columns and getters/setters
}

Где DataObjectAbstract - это простое @MappedSuperClass с версией jpa и определением равенства / хэш-кода.

У меня есть базовый класс dao, которыйвыглядит так:

public abstract class BaseDaoAbstract<T extends DataObject<K>, K extends Serializable>
        implements BaseDao<T, K>
{

    @PersistenceContext
    private EntityManager em;

    /**
     * Save a new entity. If the entity has already been persisted, then merge
     * should be called instead.
     * 
     * @param entity The transient entity to be saved.
     * @return The persisted transient entity.
     */
    @Transactional
    public T persist( T entity )
    {
        em.persist( entity );
        return entity;
    }

    /**
     * merge the changes in this detached object into the current persistent
     * context and write through to the database. This should be called to save
     * entities that already exist in the database.
     * 
     * @param entity The entity to be merged
     * @return The merged entity.
     */
    @Transactional
    public T merge( T entity )
    {
        return em.merge( entity );
    }

    // other methods like persist, delete, refresh, findByKey that all delegate to em.
}

Я определил фильтр OpenEntityManagerInView в web.xml следующим образом:

<filter>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <filter-class>
        org.springframework.orm.jpa.support.OpenEntityManagerInViewFilter
    </filter-class>
    <init-param>
        <param-name>entityManagerFactoryBeanName</param-name>
        <param-value>biEmf</param-value>
    </init-param>
</filter>

<filter-mapping>
    <filter-name>openEntityManagerInViewFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

Я недавно обновил до eclipselink 2.3.2 и Spring 3.1 и преобразовал из прокси-серверов CGLIB вПереплетение времени загрузки с aspectJ для Spring, но я не настроил LTW для eclipselink.

Проблема в этом коде, который находится в весеннем ApplicationListener, см. Комментарии.

        User user = userService.findByKey(userDetails.getKey());

        // THIS MERGE NEVER WRITES THROUGH TO THE DATABASE.
        // THIS DOESN'T WORK AS PERSIST EITHER
        user = userService.merge( user.loginSuccess() );

user.loginSuccessпросто устанавливает некоторые поля и возвращает this Я уверен, что он проходит через код, потому что я получаю лог-операторы вокруг него, и я могу установить точку останова и пройти по ней.Мой журнал postgres не показывает трафик, поступающий в postgres для слияния.

Я сохраняю другие вещи без проблем, включая пользователей в другом месте, когда они меняют свой пароль, и я знаю этот кодраньше работал.Здесь что-то явно не так?Я использую OpenEntityManagerInViewFilter неправильно?Должен ли я быть в методе @Transactional, чтобы объекты считались управляемыми?Любая помощь приветствуется.

Обновление Я попытался выполнить сброс, как было предложено праджисом.Вот код

@Transactional
public T merge( T entity )
{
    entity = em.merge( entity );
    em.flush();
    return entity;
}

в классе в com.bi.data.У меня есть это в моем весеннем конфигурационном файле приложения

<context:component-scan base-package="com.bi.controller,com.bi.data,com.bi.web" />

В моей весенней конфигурации у меня есть

<context:load-time-weaver/>
<tx:annotation-driven mode="aspectj"/>    

с aop.xml, который выглядит так:

<aspectj>
    <weaver>
        <!-- only weave classes in our application-specific packages -->
        <include within="com.bi..*"/>
    </weaver>
</aspectj>

и я получил

javax.persistence.TransactionRequiredException: 
Exception Description: No transaction is currently active

Так что что-то явно неправильно настроено, но что?

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

Ответы [ 5 ]

12 голосов
/ 31 января 2012

Попробуйте em.flush() также после em.merge(). Иногда EntityManager просто сохраняет изменения для обновления позже.

4 голосов
/ 25 июня 2015

EntityManager autocommit имеет значение false.Пожалуйста, используйте транзакцию, как показано ниже;

EntityManager em = null;
EntityTransaction t = null;
try {
    em = emf.createEntityManager();
    t = em.getTransaction();
    t.begin();
    em.merge(myTestObject);
    t.commit();
} catch (Exception e) {
    t.rollback();
    throw new RuntimeException(e.getMessage());
}finally{
    if(em != null)
        em.close();
}
2 голосов
/ 19 ноября 2013

У меня возникла сама проблема, и ее удалось решить путем изменения области действия.

@PersistenceContext(type=PersistenceContextType.EXTENDED)

ref: Управление объектами с помощью EntityManager

1 голос
/ 21 января 2014

Где методы transaction.begin() и transaction.end()?В RESOURCE_LOCAL вы должны управлять транзакцией, а не приложением ...

0 голосов
/ 09 апреля 2014

Проверьте, есть ли в вашем файле конфигурации Spring следующая строка: Это включает поддержку аннотаций @Transactional, а также убедитесь, что mode = "proxy", not mode = "aspectj" установлено

...