我正在嘗試使用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的專家。如果您需要更多信息,請告訴我。
合併()榮獲」這個錯誤是在抱怨孩子DetalheRateio被分離了,也許你在某個地方添加了一個新的實例而沒有先堅持它呢? – Gimby 2014-10-31 12:02:22
嗨,我想我在這裏分離子對象:'' List detalheRateiosBd = rateio.getDetalheRateios();'。這個對象被更新(值),我將它設置回'Rateio'對象。現在,我如何更新JPA EntityHome內的子對象? –
2014-11-07 12:22:20