2014-10-31 187 views
0

我正在嘗試使用JBoss Seam和JPA更新實體,但我面臨着分離實體這個非常常見的問題。我會解釋得更好。使用JBoss Seam的EntityHome合併JPA實體

我有兩個相關的簡單實體。這個關係映射了註釋。請看:

@Entity 
@Table(name = "RATEIO", uniqueConstraints = { @UniqueConstraint(columnNames = { "cd_projeto", "cd_tarefa", "cd_colaborador", "dt_rateio_inicial", 
     "dt_rateio_final" }) }) 
@Audited 
public class Rateio implements java.io.Serializable { 

    private static final long serialVersionUID = 1564844894403478898L; 

    private Long cdRateio; 
    private Colaborador colaborador; 
    private Tarefa tarefa; 
    private Projeto projeto; 
    private Date dtRateioInicial; 
    private Date dtRateioFinal; 

    private boolean flagAtividade; 
    private TipoRateio tipo; 
    private List<DetalheRateio> detalheRateios = new ArrayList<DetalheRateio>(0); 

    public Rateio() { 
    } 

    public Rateio(Long cdRateio, Colaborador colaborador, Projeto projeto, Date dtRateioInicial, Date dtRateioFinal) { 
     this.cdRateio = cdRateio; 
     this.colaborador = colaborador; 
     this.projeto = projeto; 
     this.dtRateioInicial = dtRateioInicial; 
     this.dtRateioFinal = dtRateioFinal; 
    } 

    public Rateio(Long cdRateio, Colaborador colaborador, Projeto projeto, Date dtRateioInicial, Date dtRateioFinal, List<DetalheRateio> detalheRateios) { 
     this.cdRateio = cdRateio; 
     this.colaborador = colaborador; 
     this.projeto = projeto; 
     this.dtRateioInicial = dtRateioInicial; 
     this.dtRateioFinal = dtRateioFinal; 
     this.detalheRateios = detalheRateios; 
    } 

    @Id 
    @SequenceGenerator(name = "sg_RATEIO_SEQ", sequenceName = "RATEIO_SEQ", allocationSize = 1) 
    @GeneratedValue(generator = "sg_RATEIO_SEQ") 
    @Column(name = "CD_RATEIO", nullable = false, scale = 0) 
    public Long getCdRateio() { 
     return this.cdRateio; 
    } 

    public void setCdRateio(Long cdRateio) { 
     this.cdRateio = cdRateio; 
    } 

    @ManyToOne(fetch = FetchType.EAGER) 
    @JoinColumn(name = "CD_COLABORADOR", nullable = true) 
    public Colaborador getColaborador() { 
     return this.colaborador; 
    } 

    public void setColaborador(Colaborador colaborador) { 
     this.colaborador = colaborador; 
    } 

    @ManyToOne(fetch = FetchType.EAGER) 
    @JoinColumn(name = "CD_PROJETO", nullable = true) 
    public Projeto getProjeto() { 
     return this.projeto; 
    } 

    public void setProjeto(Projeto projeto) { 
     this.projeto = projeto; 
    } 

    @Column(name = "DT_RATEIO_INICIAL", nullable = false) 
    public Date getDtRateioInicial() { 
     return this.dtRateioInicial; 
    } 

    public void setDtRateioInicial(Date dtRateioInicial) { 
     this.dtRateioInicial = dtRateioInicial; 
    } 

    @Column(name = "DT_RATEIO_FINAL", nullable = true) 
    public Date getDtRateioFinal() { 
     return this.dtRateioFinal; 
    } 

    public void setDtRateioFinal(Date dtRateioFinal) { 
     this.dtRateioFinal = dtRateioFinal; 
    } 

    @OneToMany(targetEntity=DetalheRateio.class, mappedBy = "rateio") 
    public List<DetalheRateio> getDetalheRateios() { 
     return this.detalheRateios; 
    } 

    public void setDetalheRateios(List<DetalheRateio> detalheRateios) { 
     this.detalheRateios = detalheRateios; 
    } 

    @ManyToOne(fetch = FetchType.EAGER) 
    @JoinColumn(name = "CD_TAREFA", nullable = true) 
    public Tarefa getTarefa() { 
     return tarefa; 
    } 

    public void setTarefa(Tarefa tarefa) { 
     this.tarefa = tarefa; 
    } 

    @Enumerated(EnumType.ORDINAL) 
    public TipoRateio getTipo() { 
     return tipo; 
    } 

    public void setTipo(TipoRateio tipo) { 
     this.tipo = tipo; 
    } 

    @Transient 
    public boolean isFlagAtividade() { 
     return flagAtividade; 
    } 

    public void setFlagAtividade(boolean flagAtividade) { 
     this.flagAtividade = flagAtividade; 
    } 
} 

我的第二個實體:

@Entity 
@Table(name = "DETALHE_RATEIO") 
@Audited 
public class DetalheRateio implements Serializable { 

    private static final long serialVersionUID = 4109721744851677683L; 

    private Long cdDetalheRateio; 
    private Rateio rateio; 
    private AtividadeOs atividadeOs; 
    private float vlPorcentagem; 

    public DetalheRateio() { 
    } 

    public DetalheRateio(Long cdDetalheRateio, Rateio rateio, AtividadeOs atividadeOs, float vlPorcentagem) { 
     this.cdDetalheRateio = cdDetalheRateio; 
     this.rateio = rateio; 
     this.atividadeOs = atividadeOs; 
     this.vlPorcentagem = vlPorcentagem; 
    } 

    @Id 
    @SequenceGenerator(name = "sg_DET_RATEIO_SEQ", sequenceName = "DET_RATEIO_SEQ", allocationSize = 1) 
    @GeneratedValue(generator = "sg_DET_RATEIO_SEQ") 
    @Column(name = "CD_DETALHE_RATEIO", nullable = false, scale = 0) 
    public Long getCdDetalheRateio() { 
     return this.cdDetalheRateio; 
    } 

    public void setCdDetalheRateio(Long cdDetalheRateio) { 
     this.cdDetalheRateio = cdDetalheRateio; 
    } 

    @ManyToOne(targetEntity = Rateio.class, cascade=CascadeType.ALL) 
    @JoinColumn(name = "CD_RATEIO", nullable = false) 
    public Rateio getRateio() { 
     return this.rateio; 
    } 

    public void setRateio(Rateio rateio) { 
     this.rateio = rateio; 
    } 

    @ManyToOne(fetch = FetchType.LAZY) 
    @JoinColumn(name = "CD_ATIVIDADE_OS", nullable = false) 
    public AtividadeOs getAtividadeOs() { 
     return this.atividadeOs; 
    } 

    public void setAtividadeOs(AtividadeOs atividadeOs) { 
     this.atividadeOs = atividadeOs; 
    } 

    @Column(name = "VL_PORCENTAGEM", nullable = false, precision = 19, scale = 2, columnDefinition = "NUMBER") 
    public float getVlPorcentagem() { 
     return this.vlPorcentagem; 
    } 

    public void setVlPorcentagem(float vlPorcentagem) { 
     this.vlPorcentagem = vlPorcentagem; 
    } 
} 

OK,到目前爲止,一切都很好。現在,我有一個從JBoss Seam擴展EntityHome的類,並且這個類試圖更新我的實體。這裏是代碼:

@Name("rateioHome") 
public class RateioHome extends EntityHome<Rateio> { 

    @SuppressWarnings("unchecked") 
    @Override 
    @Transactional 
    public String update() { 
     String result = null; 
     boolean valid = true; 
     if (getPercentualTotal().doubleValue() < 100 || instance.getDetalheRateios().size() < 2) { 
      facesMessages.addFromResourceBundle(Severity.ERROR, "rateio.erro.atividade"); 
      valid = false; 
     } 

     if (instance.getDtRateioFinal() != null && instance.getDtRateioFinal().before(instance.getDtRateioInicial())) { 
      facesMessages.addFromResourceBundle(Severity.ERROR, "rateio.erro.data"); 
      valid = false; 
     } 

     if (valid) { 
      Rateio rateio = super.find(); 
      List<DetalheRateio> detalheRateios = getInstance().getDetalheRateios(); 
      List<DetalheRateio> detalheRateiosBd = rateio.getDetalheRateios(); 
      Map<Long, DetalheRateio> mapDetalheRateio = new HashMap<Long, DetalheRateio>(); 

      if (detalheRateiosBd != null && detalheRateiosBd.size() > 0) { 

       Rateio _rateio = getInstance(); 

       for (DetalheRateio item : detalheRateiosBd) { 

        if (item.getCdDetalheRateio() != null) { 

         item.getRateio().setDtRateioInicial(_rateio.getDtRateioInicial()); 
         item.getRateio().setDtRateioFinal(_rateio.getDtRateioFinal()); 
         mapDetalheRateio.put(item.getCdDetalheRateio(), item); 
        } 
       } 
      } 

      List<DetalheRateio> novos = (List<DetalheRateio>) CollectionUtils.subtract(detalheRateios, detalheRateiosBd); 
      List<DetalheRateio> excluidos = (List<DetalheRateio>) CollectionUtils.subtract(detalheRateiosBd, detalheRateios); 

      detalheRateiosBd.removeAll(excluidos); 
      detalheRateiosBd.addAll(novos); 

      setInstance(rateio); 
      result = super.update(); 
      getEntityManager().flush(); 
     } 
     return result; 
    } 
} 

現在,問題。當我TR更新上線實體:result = super.update();,我面對此異常:

javax.persistence.PersistenceException: org.hibernate.PersistentObjectException: detached entity passed to persist: br.com.entity.DetalheRateio

我試圖找到一種方法merge上EntityHome,但我沒有發現任何東西。這是我必須支持的遺留代碼,我不是JBoss Seam的專家。如果您需要更多信息,請告訴我。

+0

合併()榮獲」這個錯誤是在抱怨孩子DetalheRateio被分離了,也許你在某個地方添加了一個新的實例而沒有先堅持它呢? – Gimby 2014-10-31 12:02:22

+0

嗨,我想我在這裏分離子對象:'' List detalheRateiosBd = rateio.getDetalheRateios();'。這個對象被更新(值),我將它設置回'Rateio'對象。現在,我如何更新JPA EntityHome內的子對象? – 2014-11-07 12:22:20

回答

0

剛剛找到解決方案。在更新對象Rateio之前,對象列表DetalheRateio必須合併(持續)。我因子評分的對象DetalheRateio爲Rateio孩子將自動JPA堅持」我的錯所以,我合併手動每個DetalheRateio對象的,我用來做這snipet:。

List<DetalheRateio> novos = (List<DetalheRateio>) CollectionUtils.subtract(detalheRateios, detalheRateiosBd); 
List<DetalheRateio> excluidos = (List<DetalheRateio>) CollectionUtils.subtract(detalheRateiosBd, detalheRateios); 
List<DetalheRateio> detalhesAlterados = new ArrayList<DetalheRateio>(); 

detalheRateiosBd.removeAll(excluidos); 
detalheRateiosBd.addAll(novos); 

for (DetalheRateio detalheExcluido : excluidos) { 
    getEntityManager().remove(detalheExcluido); 
} 

for (DetalheRateio dRateio : detalheRateios) { 
    detalhesAlterados.add(getEntityManager().merge(dRateio)); 
} 

rateio.setDetalheRateios(detalhesAlterados); 
setInstance(rateio); 
result = super.update(); 
getEntityManager().flush();