java - spring jpa安装程序无法更新数据库

标签 java hibernate spring jpa

我是一个长期读者,第一次发帖。

本质上,我正在编写一个基于 spring 3 showcase 的网络应用程序.

随后我插入了基于 this tutorial 写入数据库的功能.

测试所有 ORM 内容的集成测试看起来很满意,但是当我将项目部署到 tomcat 并尝试通过网站进行数据库更新时,数据库没有更新!

配置:

appContext.xml

<!-- holding properties for database connectivity / -->
<context:property-placeholder location="classpath:jdbc.properties" />
<!-- enabling annotation driven configuration / -->
<context:annotation-config />
<context:component-scan base-package="wcpackage" />
<tx:annotation-driven transaction-manager="transactionManager" />

<bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />

<bean id="dataSource"
    class="org.springframework.jdbc.datasource.DriverManagerDataSource"
    p:driverClassName="${jdbc.driverClassName}" p:url="${jdbc.url}" />

<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager"
    p:entityManagerFactory-ref="entityManagerFactory" />

<bean id="entityManagerFactory"
    class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
    p:dataSource-ref="dataSource" p:jpaVendorAdapter-ref="jpaAdapter">
    <property name="loadTimeWeaver">
        <bean
            class="org.springframework.instrument.classloading.InstrumentationLoadTimeWeaver" />
    </property>
    <property name="persistenceUnitName" value="wctemplatePU"></property>
</bean>

<bean id="jpaAdapter"
    class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter"
    p:showSql="${jpa.showSql}" />

jdbc.属性

jpa.database=gm
jpa.showSql=true
jdbc.driverClassName=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/gm?user=gmuser&password=gmuser

持久性.xml

<persistence xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://java.sun.com/xml/ns/persistence">

<persistence-unit name="wctemplatePU" transaction-type="RESOURCE_LOCAL">
    <properties>
        <property name="hibernate.hbm2ddl.auto" value="validate" />
    </properties>
</persistence-unit>

应提交到数据库的代码

    @Transactional(propagation=Propagation.REQUIRED, rollbackFor=Exception.class)
protected Sequence persistGame(PGNGame game) {
    Sequence seq = new Sequence();
    seq.setType(SequenceType.GAME);
    seq.setEvent(game.getTag("Event"));
    seq.setSite(game.getTag("Site"));

            ...

    sequenceDao.persist(seq);

    return seq;
}

SequenceDao 本质上是

public abstract class JpaDao<K, E> implements Dao<K, E> {
protected Class<E> entityClass;

@PersistenceContext
protected EntityManager entityManager;

public JpaDao() {
    ParameterizedType genericSuperclass = (ParameterizedType) getClass().getGenericSuperclass();
    this.entityClass = (Class<E>) genericSuperclass.getActualTypeArguments()[1];
}

public void persist(E entity) { entityManager.persist(entity); }

public void remove(E entity) { entityManager.remove(entity); }

public E findById(K id) { return entityManager.find(entityClass, id); }
}

我在控制台/日志中没有看到任何类型的错误。我做错了什么?

当我将 hibernate 日志记录设置为“DEBUG”时,我注意到一件事

DEBUG:org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler - Creating new EntityManager for shared EntityManager invocation
    DEBUG: org.hibernate.impl.SessionImpl - opened session at timestamp: 12975078282
    DEBUG: org.hibernate.event.def.AbstractSaveEventListener - delaying identity-insert due to no transaction in progress
    DEBUG: org.hibernate.event.def.AbstractSaveEventListener - delaying identity-insert due to no transaction in progress
    DEBUG: org.springframework.orm.jpa.EntityManagerFactoryUtils - Closing JPA EntityManager

跟踪日志是

DEBUG:org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler - Creating new EntityManager for shared EntityManager invocation
DEBUG: org.hibernate.impl.SessionImpl - opened session at timestamp: 12975085724
TRACE: org.hibernate.engine.IdentifierValue - id unsaved-value: 0
TRACE: org.hibernate.event.def.AbstractSaveEventListener - transient instance of: net.samuelbergin.gm.model.Player
TRACE: org.hibernate.event.def.DefaultPersistEventListener - saving transient instance
TRACE: org.hibernate.event.def.AbstractSaveEventListener - saving [net.samuelbergin.gm.model.Player#<null>]
TRACE: org.hibernate.engine.Cascade - processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Player
TRACE: org.hibernate.engine.Cascade - done processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Player
TRACE: org.hibernate.event.def.WrapVisitor - Wrapped collection in role: net.samuelbergin.gm.model.Player.sequenceList
DEBUG: org.hibernate.event.def.AbstractSaveEventListener - delaying identity-insert due to no transaction in progress
TRACE: org.hibernate.engine.Cascade - processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Player
TRACE: org.hibernate.engine.Cascade - cascade ACTION_PERSIST_SKIPLAZY for collection: net.samuelbergin.gm.model.Player.sequenceList
TRACE: org.hibernate.engine.CascadingAction - cascading to persist: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.engine.IdentifierValue - id unsaved-value: 0
TRACE: org.hibernate.event.def.AbstractSaveEventListener - transient instance of: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.event.def.DefaultPersistEventListener - saving transient instance
TRACE: org.hibernate.event.def.AbstractSaveEventListener - saving [net.samuelbergin.gm.model.Sequence#<null>]
TRACE: org.hibernate.engine.Cascade - processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.engine.Cascade - done processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.event.def.WrapVisitor - Wrapped collection in role: net.samuelbergin.gm.model.Sequence.commentList
TRACE: org.hibernate.engine.IdentifierValue - id unsaved-value: 0
DEBUG: org.hibernate.event.def.AbstractSaveEventListener - delaying identity-insert due to no transaction in progress
TRACE: org.hibernate.engine.Cascade - processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.engine.Cascade - cascade ACTION_PERSIST_SKIPLAZY for collection: net.samuelbergin.gm.model.Sequence.commentList
TRACE: org.hibernate.engine.Cascade - done cascade ACTION_PERSIST_SKIPLAZY for collection: net.samuelbergin.gm.model.Sequence.commentList
TRACE: org.hibernate.engine.Cascade - done processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Sequence
TRACE: org.hibernate.engine.Cascade - done cascade ACTION_PERSIST_SKIPLAZY for collection: net.samuelbergin.gm.model.Player.sequenceList
TRACE: org.hibernate.engine.Cascade - done processing cascade ACTION_PERSIST_SKIPLAZY for: net.samuelbergin.gm.model.Player
DEBUG: org.springframework.orm.jpa.EntityManagerFactoryUtils - Closing JPA EntityManager
TRACE: org.hibernate.impl.SessionImpl - closing session

实体定义

@Entity
@Table(name="sequences")
public class Sequence {
    @Id
    @GeneratedValue
    private int id;
    @Enumerated(EnumType.STRING)
    private SequenceType type;
    private String event;
    private String site;
    @Temporal(TemporalType.DATE)
    private Date date;
    private int round;
    private String result;
    private String eco;
    private String description;
    private String moves;
    @ManyToOne
    private Player whitePlayer;
    @ManyToOne
    private Player blackPlayer;
    @OneToMany(cascade={CascadeType.ALL})
    @JoinTable(name="sequenceComments",
            joinColumns={@JoinColumn(name="sequence_fk")},
            inverseJoinColumns={@JoinColumn(name="comment_fk")})
    private List<Comment> commentList = new ArrayList<Comment>();
        ...

Controller

@Controller
@RequestMapping("/fileupload")
public class FileUploadController {

    private static Logger logger = LoggerFactory.getLogger(FileUploadController.class);

    @Autowired private SequenceService sequenceService;

    @RequestMapping(method=RequestMethod.POST)
    public void processUpload(@RequestParam MultipartFile file, HttpServletRequest request, Model model) throws IOException {
        //TODO: parse pgn
        List<PGNGame> pgnGameList;
        String filename = file.getOriginalFilename();
        try {
            pgnGameList = PGNProcessor.parse(file.getInputStream());

            sequenceService.createSequences(pgnGameList);

            String message = "File '" + filename + "' uploaded successfully";
            model.addAttribute("message", new Message(MessageType.success, message));

        } catch (PGNProcessorException ppe) {
            logger.error("Failed to process "+filename, ppe);
            model.addAttribute("message", new Message(MessageType.error, "Failed to process "+filename+".  Reason: "+ppe.getMessage()));
        }
        // else whole html pg is rendered instead of just a div to update
        model.addAttribute("ajaxRequest", AjaxUtils.isAjaxUploadRequest(request));  
    }
}

服务

public class JpaSequenceService implements SequenceService {

    ...

public List<Sequence> createSequences(List<PGNGame> gameList) {
    List<Sequence> seqList = new ArrayList<Sequence>();
    for (PGNGame game : gameList) {
        Sequence seq = persistGame(game);
        seqList.add(seq);
    }
    return seqList;
}

@Transactional(propagation=Propagation.REQUIRED, rollbackFor=Exception.class)
protected Sequence persistGame(PGNGame game) {
        /*...as above...*/
}

最佳答案

@Transactional仅当其放置在从其他对象(而不是同一对象内)调用的公共(public)方法上时才有效。

通过在对象周围创建代理来处理事务。来自外部的调用被代理拦截,从而启动事务。同一对象内的调用不会被代理拦截。

所以在你的情况下:

@Transactional(propagation=Propagation.REQUIRED, rollbackFor=Exception.class)
public List<Sequence> createSequences(List<PGNGame> gameList) {

请注意,将工作单元包装在事务中通常是一个好习惯。即使您的代码有效,它也会不必要地打开和关闭多个事务。除非你想单独坚持每个游戏。如果是这种情况,则要么迭代 Controller 中的列表,要么创建一个新类来保存游戏。

正如 Augusto 所说,如果您使用 proxy-target-class="true"对于您的代理,您可以进行内部调用来启动交易。不过,我不鼓励这样做,因为您正在变得依赖于此配置选项,并且如果您将来因其他原因进行更改,您可能会浪费时间查找现有功能停止工作的原因。

关于java - spring jpa安装程序无法更新数据库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4977509/

相关文章:

java - 在场景渲染之前在 Initialize 中调用查找

java - 在 @DynamoDBTable 中创建 List<MyObject> 而不使用 dynamodbmarshalling 的替代方法(已弃用)

hibernate - 什么会导致Hibernate InvalidStateException

java - 通过字段 'clientRepository' 表示不满足的依赖关系;

java - CXF服务协助

java - SimpleJaxWsServiceExporter 和 wssecurity

java - 从 Java 7 开始,字符串内部池

java - 某些第三方库的 Logback 默认为 Debug模式

java - 没有为依赖项 : expected at least 1 bean which qualifies as autowire candidate for this dependency 找到 UserRepository 类型的合格 bean

java.lang.NoClassDefFoundError : org/hibernate/type/StandardBasicTypes - What could be the reason? 错误