Solucionado (ver solução)
Solucionado
(ver solução)
2
respostas

org.hibernate.PersistentObjectException: detached entity passed to persist: br.com.alura.jpa.modelo.Conta (?)

A tabela de conta não é criada e aparece essa exceção e não sei o que pode estar errado.

package br.com.alura.jpa.teste;

import br.com.alura.jpa.modelo.Conta;

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;


public class CriaConta {

    public static void main(String[] args) {

        EntityManagerFactory emf = Persistence.createEntityManagerFactory("alura");
        EntityManager em = emf.createEntityManager();

        Conta c = new Conta();
        c.setTitular("Fernanda");
        c.setAgencia(3339);
        c.setNumero(1234);

        em.getTransaction().begin();

        em.persist(c);

        em.getTransaction().commit();

    }

}

o persistence:

<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
        http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"
    version="2.0">

    <persistence-unit name= "alura">
        <provider>org.hibernate.jpa.HibernatePersistenceProvider</provider>
        <class>br.com.alura.jpa.modelo.Conta</class>

        <properties>
            <property name="javax.persistence.jdbc.driver" value="com.mysql.jdbc.Driver" />
            <property name="javax.persistence.jdbc.url" value="jdbc:mysql://localhost/alura_jpa" />
            <property name="javax.persistence.jdbc.user" value="root" />
            <property name="javax.persistence.jdbc.password" value="" />

            <property name="hibernate.dialect" value="org.hibernate.dialect.MySQL5InnoDBDialect" />
            <property name="hibernate.show_sql" value="true" />
            <property name="hibernate.format_sql" value="true" />

            <property name="hibernate.hbm2ddl.auto" value="update" />

        </properties>
    </persistence-unit>
</persistence>

e a class Conta:

package br.com.alura.jpa.modelo;
import javax.persistence.Basic;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;

@Entity
public class Conta {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Basic(optional=false)
    @Column(name="id",unique=true,nullable=false)
    private Integer agencia;
    private    Integer numero;
    private String titular;
    private Long Id;
    private Double saldo;

    public Integer getAgencia() {
        return agencia;
    }
    public void setAgencia(Integer agencia) {
        this.agencia = agencia;
    }
    public Integer getNumero() {
        return numero;
    }
    public void setNumero(Integer numero) {
        this.numero = numero;
    }
    public String getTitular() {
        return titular;
    }
    public void setTitular(String titular) {
        this.titular = titular;
    }
    public Long getId() {
        return Id;
    }
    public void setId(Long Id) {
        this.Id = Id;
    }
    public Double getSaldo() {
        return saldo;
    }
    public void setSaldo(Double saldo) {
        this.saldo = saldo;
    }
}
2 respostas
solução!

Olá Fernanda, tudo bem com você?

Vou dar uma explicação apontando o que causou o problema, mas se não quiser saber dos detalhes pode pular para o final onde eu dou uma solução diretamente! :)

Veja que aqui você está definindo agencia como chave-primária com a estratégia GenerationType.AUTO. Isso quer dizer que nós estamos atribuindo um valor à chave-primária agencia DUAS vezes!

Uma no código da Conta: c.setAgencia(3339);

E outra através do próprio banco: @GeneratedValue(strategy = GenerationType.AUTO)

Quando o banco vai atribuir o valor à chave-primária agencia e ele descobre que já foi atribuído um valor à ela anteriormente, ele assume que o objeto já existia anteriormente, mas que agora não está mais conectado ao EntityManager (é o que chamamos de detached), daí a exceção! Ou seja, basta remover o @GeneratedValue(strategy = GenerationType.AUTO) que resolvemos a exceção!

Mas só isso não resolve todos os problemas, agora temos outro problema... veja que a Conta tem DOIS ids agora! Aqui na anotação @Column(name="id",unique=true,nullable=false) você está declarando que agencia terá o nome id quando for registrado no banco de dados. Mas perceba que temos um outro atributo da classe Conta com o nome id também, isso faz com que o banco tente registrar duas colunas com o mesmo e nome e joga outra exceção! Podemos resolver isso apagando o atributo id da Conta... mas você como agora não faz sentido ter a agencia como chave-primária? Seria como se cada agência estivesse limitada gerenciar apenas uma conta, o que não faz sentido no mundo real.

Enfim, para resolver o problema não precisamos alterar nenhuma anotação. Apenas troque a agencia e o id de lugar:

@Id
@GeneratedValue(strategy = GenerationType.AUTO)
@Basic(optional=false)
@Column(name="id",unique=true,nullable=false)
private Long id;
private Integer agencia;
private Integer numero;
private String titular;
private Double saldo;

Dessa forma, será o banco que atribuirá o valor ao id e como agencia terá o mesmo nome no banco também, não teremos duas colunas chamadas id.

Espero ter ajudado. Se ficou alguma dúvida é só avisar!

Bons estudos!!

Consegui entender perfeitamente o que estava errando, muito obrigada pela explicação!