2

我有一个现有的 java-JSF 项目,我被要求添加持久性,使用 JPA 和 Hibernate,当然还有 MVC。我使用 Postgres 作为 DBMS,使用 Netbeans 作为我的 IDE。我做了很多研究,结果比开始之前更加困惑,所以这就是我到目前为止所得到的:

我的项目需要是一个工作列表网站,用户可以在其中注册然后申请工作机会。但我只是坚持注册用户。我的视图有一个使用 JSF 构建的 register.xhtml 页面,我的模型包中有一个 User.java 类,我的控制器中有一个 UserController bean。register.xhtml 调用 UserController bean 中的 register() 方法,该方法应该保留 User 对象。

我已经像这样配置了我的 persistence.xml:

<persistence version="2.0" 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 http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">  
<persistence-unit name="TP5PU" transaction-type="RESOURCE_LOCAL">  
<provider>org.hibernate.ejb.HibernatePersistence</provider>  
<non-jta-data-source>tp5</non-jta-data-source>  
<class>model.User</class>  
<exclude-unlisted-classes>true</exclude-unlisted-classes>  
<properties>  
  <property name="configLocation" value="WEB-INF/classes/hibernate.cfg.xml"/>  
  <property name="hibernate.dialect" value="org.hibernate.dialect.PostgreSQLDialect"/>  
  <property name="hibernate.hbm2ddl.auto" value="update"/>  
  <property name="hibernate.connection.username" value="postgres"/>  
  <property name="hibernate.connection.password" value="postgres"/>  
  <property name="hibernate.connection.url" value="jdbc:postgresql://localhost:5432/pootp5"/>  
</properties>  
</persistence-unit>  
</persistence>  

我还创建了一个 PersistenceManager.java 类,它是一个单例,其中还有一个单例 EntityManagerFactory:

import javax.persistence.*;

public class PersistenceManager {

private EntityManagerFactory emf = null;
private EntityManager em = null;
private static PersistenceManager instance = null;

private PersistenceManager(){
}

public static PersistenceManager getInstance() {
    if (instance==null){
        instance=new PersistenceManager();
    }
    return instance;
}

public EntityManagerFactory getEmf() {
    if (emf == null){
        this.setEmf(Persistence.createEntityManagerFactory("TP5PU"));
    }
        return emf;
}

public EntityManager getEm() {
    this.setEm(this.getEmf().createEntityManager());
    return em;
}

public void setEmf(EntityManagerFactory emf) {
    this.emf = emf;
}


public void setEm(EntityManager em) {
    this.em = em;
}
}

现在,我的 UserController bean 有以下代码:

import javax.faces.bean.ManagedBean;
import javax.faces.bean.SessionScoped;
import java.io.Serializable;
import javax.persistence.EntityTransaction;
import model.PersistenceManager;
import model.Usuario;

@ManagedBean (name="userController")
@SessionScoped
public class UserController implements Serializable{

private String username;
private String password;
private String name;
private String lastname;
private String resume;
private PersistenceManager pm;
private EntityTransaction tx;

public UserController() {
    pm = PersistenceManager.getInstance();
    tx = pm.getEm().getTransaction();
}


public String register() {
    User user = new User(this.getUsername(), this.getPassword(), this.getName(), this.getLastname(), this.getResume());
    try {
        getPm().getEmf();
        getTx().begin();
        getPm().getEm().persist(user);
        getTx().commit();
        getPm().getEm().close();
        return "registered";
    }
    catch (org.hibernate.exception.ConstraintViolationException e1){   
        if (getTx().isActive()){
            getTx().rollback();}
        getPm().getEm().close();
        return "duplicateuser";
    }
    catch (Exception e2){   
        if (getTx().isActive()){
            getTx().rollback();}            
        getPm().getEm().close();
        return "registrationerror";
    }
}

//all getters & setters
}

我所有的 User.java 类都有一堆属性和 getter/setter 方法:

import java.io.Serializable;
import javax.persistence.*;

@Entity
public class User implements Serializable {

@Id
@GeneratedValue(strategy=GenerationType.AUTO)
@Column(updatable=false)
protected long userID;
@Column(unique=true)
private String username;
private String password;
private String name;
private String lastname;
private String resume;

//and all the getters & setters needed for each attribute.

register.xhtml 页面有一个注册表单(我不包括在这里,因为它只是一团糟,所有 i18n 键和东西)当按下“注册”按钮时调用 UserController bean:

如果 register() 方法返回“已注册”,则有一个导航规则将用户带到应用程序登录。如果它返回“duplicateuser”,则将其带回 register.xhtml,如果它抛出不同的错误消息,它也会将用户带回 register.xhtml(是的,这些涉及的逻辑并不多,我需要保持简单)。

问题是,我创建了我的用户表,但没有用户保存在其中,并且我没有收到任何错误消息、异常或任何东西。调试时,我在 UserController bean 中的 EntityTransaction 对象将事务显示为 begin=true 然后committed=true,所以一切似乎都在工作。但是我去检查我的数据库并创建了表,但其中没有用户:(

另外,我想知道当将 PersistenceManager 作为单例并将所有持久性逻辑粘贴在 UserController bean 中时,我是否以正确的方式进行。如果有人能指导我一点,我将非常感激!:)

4

1 回答 1

1

您的 getEM() 方法在每次调用时都会创建一个新的 EntityManager。这意味着您用来启动事务的实体管理器与您调用persist(user)的实体管理器不同;上。因此事务提交,但不绑定到包含您的更改的 EM。

使用静态 PM 来保存 EntityManager 是个坏主意,因为 EntityManager 本身不是线程安全的。使用获取EntityManagers可能很好,但是注册和其他方法需要直接持有entityManager,完成后应该关闭它。PM 还应该有一个清理方法,在关闭或取消部署时关闭其工厂。

于 2012-11-30T16:39:53.353 回答