java - EntityManager.merge 插入重复实体

标签 java hibernate persistence sqlite entitymanager

我有一个非常特殊的设置:我用 Java 创建所有类,在我的应用程序中连接它们(几个多对一关系)。

然后,我想遍历我的对象并将它们保存到数据库中。有时,一个对象已经在数据库中,那么它不应该被再次持久化。

我正确地实现了 hashCode() 和 equals() 方法,但是我的 em.merge() 仍然插入了对象。

再次:

我创建了一些对象,即我创建了一些球员并设置了他们所在的球队。团队在 Java 中可能是不同的对象,但根据它们的“等于”方法,它们是相同的。所以如果我保存一个球员,球队应该相应地保存(有效),但如果球队存在于数据库中(根据 equals-method),它不应该再次插入,但应该设置关系,当然。

我做错了什么?需要更多信息?

    private static void saveModels(final Set<?> models) {
    EntityManagerFactory factory = null;

    factory = Persistence.createEntityManagerFactory("sqlite");

    EntityManager manager = factory.createEntityManager();

    manager.getTransaction().begin();

    for (Object object : models) {
        manager.merge(object);
    }

    manager.getTransaction().commit();

    manager.close();
    factory.close();
}

编辑

@Entity
public class Team {

    private long id;
    private String description;

    @Id
    @GeneratedValue
    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getDescription() {
        return description;
    }

    public void setDescription(String description) {
        this.description= description;
    }

    /*
     * (non-Javadoc)
     * 
     * @see java.lang.Object#hashCode()
     */
    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + description.length();
        return result;
    }

    /*
     * (non-Javadoc)
     * 
     * @see java.lang.Object#equals(java.lang.Object)
     */
    @Override
    public boolean equals(final Object obj) {
        if (this == obj) {
            return true;
        }
        if (obj == null) {
            return false;
        }
        if (getClass() != obj.getClass()) {
            return false;
        }
        Team other = (Team) obj;
        if (!description.equals(other.getDescription())) {
            return false;
        }
        return true;
    }
}


@Entity
public class Player {
    private long id;
    private Team team;
    private String name;

    @Id
    @GeneratedValue
    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    @ManyToOne(cascade = { CascadeType.PERSIST, CascadeType.MERGE }, targetEntity = Team.class)
    @JoinColumn(name = "team_id")
    public Team getTeam() {
        return team;
    }

    public void setTeam(Team team) {
        this.team = team;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }


    @Override
    public int hashCode() {
        return name.length();
    }

    @Override
    public boolean equals(Object obj) {
        if (!(obj instanceof Player)) {
            return false;
        }

        Player other = (Player) obj;
        return other.getName().equals(name);
    }
}

最佳答案

JPA 使用@Id 字段进行合并,它不会使用 equals 和 hashCode 方法来检查数据库中是否已存在实体。

在团队中添加一个@OneToMany 映射,就像 Bozho 建议的那样,尽管我会这样做。

@OneToMany(cascade = { CascadeType.ALL })
private List<Player> players = new ArrayList<Player>();

public void addPlayer(Player player) {
    player.setTeam(this);
    players.add(player);
}

public Collection<Player> getPlayers() {
    return new ArrayList<Player>(this.players);
}

当您同时合并大量实体时,我假设您正在从 CSV 或其他内容中进行批量导入。然后,不要为 CSV 中的每一行创建一个新的团队/球员,而是保留一个以名称为键的球队 map ,并将球员添加到相关球队。

所以,而不是

Team t = new Team();
t.setName(teamName)
Player p = new Player();
p.setName(playerName);
p.setTeam(t);

Map<String, Team> teams = new HashMap<String,Team>();

...

if (!teams.containsKey(teamName)) {
    Team t = new Team();
    t.setDescription(teamName);
    teams.put(teamName, t)
}

Player p = new Player();
p.setName(p);
teams.get(teamName).addPlayer(p);

...

saveModels(teams.values());

关于java - EntityManager.merge 插入重复实体,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/1975708/

相关文章:

java - 在 Java-EE 中透明地管理每个请求事务

java - 多部分文件字节表示

java - Selenium Grid + Maven + TestNG + 范围报告 + 并行测试

SQL 查询中的 Hibernate ManyToOne 关联和连接表

java - UniqueResult() 方法在 hibernate 条件查询中返回一个新对象吗?

java - 持久模型到域模型的映射,无需公开域对象属性

java - 为什么 JMenu 不总是在最前面?

java - 动态报表 + Hibernate

excel - 如何在 Excel 工作簿中存储信息

perl - 如何在 Perl 中存储引用列表?