代码之家  ›  专栏  ›  技术社区  ›  user1778855

用@Embeddable保存实体时的PropertyAccessException

  •  0
  • user1778855  · 技术社区  · 6 年前

    map a many-to-many association with extra columns tutorial 但不是很成功。

    所以我有以下实体。。。

    @Data
    @Entity
    @Table(name = "PEOPLE")
    public class People implements Serializable {
    
        @Id
        @SequenceGenerator(name = "people", sequenceName = "people_id_seq", allocationSize = 1)
        @GeneratedValue(strategy = GenerationType.AUTO, generator = "people")
        private long peopleId;
    
        private String peopleName;
    
        @ToString.Exclude
        @EqualsAndHashCode.Exclude
        @OneToMany(
                mappedBy = "people", 
                cascade = CascadeType.ALL, 
                orphanRemoval = true
        )
        private List<PeopleStats> peopleStats;
    
        public void addStats(Stats stats) {
    
            if (this.peopleStats == null) {
                this.peopleStats = new ArrayList<>();
            }
    
            PoepleStats pStats = PoepleStats.builder().people(this).stats(stats).build();
    
            this.peopleStats.add(pStats);
        }
    }
    
    @Data
    @Entity
    @Table(name = "STATS")
    public class Stats implements Serializable {
    
        @Id
        @SequenceGenerator(name = "stats", sequenceName = "stats_id_seq", allocationSize = 1)
        @GeneratedValue(strategy = GenerationType.AUTO, generator = "stats")
        private long statsId;
    
        private String statsName;
        private String statsDescription;
    
    }
    
    @Data
    @Entity
    @Table(name = "PEOPLE_STATS")
    public class PeopleStats implements Serializable {
    
        @EmbeddedId
        private PeopleStatsId peopleStatsId;
    
        @ManyToOne(fetch = FetchType.LAZY)
        @MapsId("peopleId")
        private People people;
    
        @ManyToOne(fetch = FetchType.LAZY)
        @MapsId("statsId")
        private Stats stats;
    
        private long value;
    
    }
    
    @Data
    @Embeddable
    @EqualsAndHashCode
    public class PeopleStatsId implements Serializable {
    
        // Putting @Column(name = "people_id") or not doesn't seem to have any effect
        private long peopleId;
        // Same goes for this
        private long statsId;
    
    }
    

    然后进行以下单元测试。。

    @RunWith(SpringRunner.class)
    @DataJpaTest
    public class PeopleRepositoryTest {
    
        @Autowired
        private TestEntityManager entityManager;
    
    
        @Test
        public void testSavePeople() {
    
            // People object created
            people.addStats(Stats.builder().statsId(new Long(1)).statsName("a").statsDescription("b").build());
    
            this.entityManager.persistAndFlush(people);
        }
    }
    

    Hibernate: create table people_stats (value bigint not null, people_people_id bigint not null, stats_stats_id bigint not null, primary key (people_people_id, stats_stats_id))
    

    这是stacktrace。。

    javax.persistence.PersistenceException异常: org.hibernate.PropertyAccessException网站:无法设置字段值 1 com.sample.shared网站.实体.PeopleStatsId.peopleId]设置者 com.sample.shared网站.实体.PeopleStatsId.peopleId在 org.hibernate.internal公司.前ceptionConverterImpl.convert转换(前java:149) org.hibernate.internal公司.前ceptionConverterImpl.convert转换(前java:157) 在 org.hibernate.internal公司.前ceptionConverterImpl.convert转换

    我碰到这个 similar issue ,有解决方案但不起作用。在尝试了第一个解决方案之后,即创建 new PeopleStatsId @EmbeddedId ,我也犯了同样的错误。

    有人能带我走吗?谢谢。

    github .

    更新2

    public void addStats(Stats stats) {
    
            if (this.peopleStats == null) {
                this.peopleStats = new ArrayList<>();
            }
    
            PeopleStats pStats = PeopleStats.builder().peopleStatsId(new PeopleStatsId()).people(this).stats(stats).build();
    
            this.peopleStats.add(pStats);
        }
    

    它现在正在抛出分离实体错误。

    原因:org.hibernate.persistentobject异常:分离的实体 org.hibernate.event网站.内部.DefaultPersistEventListener.onPersist(默认java:124的persistentEventListener.java) ... 68个以上

    更新3 :

    我变了 CascadeType.ALL MERGE 它似乎解决了这个问题,但我不知道为什么。我甚至删除了更新2中关于 .peopleStatsId(new PeopleStatsId()) 它也能工作。现在我更困惑了。

    在人民课上:

        @OneToMany(
                mappedBy = "people", 
                cascade = CascadeType.MERGE, 
                orphanRemoval = true
        )
        private List<PeopleStats> peopleStats;
    
        public void addStats(Stats stats) {
    
            if (this.peopleStats == null) {
                this.peopleStats = new ArrayList<>();
            }
    
            PeopleStats pStats = PeopleStats.builder().people(this).stats(stats).build();
    
            this.peopleStats.add(pStats);
        }
    
    2 回复  |  直到 6 年前
        1
  •  0
  •   Guts    6 年前

    你需要在PeopleStats类中设置peopleStatsId。所以换行:

    @EmbeddedId
    private PeopleStatsId peopleStatsId;
    

    对此:

    @EmbeddedId
    private PeopleStatsId peopleStatsId = new PeopleStatsId();
    

    Hibernate正在尝试设置PeopleStatsId的字段,但该实例等于null,因此引发了NullPointerException。

        2
  •  0
  •   user1778855    6 年前

    @K.Nicholas

    @K、 尼古拉斯 关于人员和统计的字段设置是对的,但我一开始不太清楚。不管怎样,我后来才知道。

    People 上课。

    @Data
    @Entity
    @Builder
    @Table(name = "PEOPLE")
    public class People implements Serializable {
    
        private static final long serialVersionUID = 1L;
    
        @Id
        @SequenceGenerator(name = "people", sequenceName = "people_id_seq", allocationSize = 1)
        @GeneratedValue(strategy = GenerationType.AUTO, generator = "people")
        private long peopleId;
    
        private String peopleName;
    
        @ToString.Exclude
        @EqualsAndHashCode.Exclude
        @OneToMany(
                mappedBy = "people", 
                cascade = CascadeType.ALL, 
                orphanRemoval = true
        )
        private List<PeopleStats> peopleStats;
    
        // Maintain the state of objects association
        public void addStats(Stats stats) {
    
            if (this.peopleStats == null) {
                this.peopleStats = new ArrayList<>();
            }
    
            // Important to ensure that a new instance of PeopleStatsId is being passed into the field
            // otherwise, PropertyAccessException will be encountered
            PeopleStats pStats = PeopleStats.builder()
                                    .people(this)
                                    .stats(stats)
                                    .peopleStatsId(new PeopleStatsId(this.getPeopleId(), stats.getStatsId()))
                                    .build();
    
            this.peopleStats.add(pStats);
        }
    
    }
    

    注意中的评论 addStats PeopleStatsId 对象来初始化 人状态

    我还提到我以前遇到过分离实体的问题。这是因为我试图在Stats id字段中设置它,而不是必需的。

    根据 Hibernate Guide

    实体具有关联的标识符,但不再与持久性上下文关联(通常是因为持久性 上下文已关闭或实例已从上下文中逐出)

    在我的帖子中,我试图给人们设置统计数据,然后坚持下去。

    @Test
        public void testSavePeople() {
    
            // People object created
            people.addStats(Stats.builder().statsId(new Long(1)).statsName("a").statsDescription("b").build());
    
            this.entityManager.persistAndFlush(people);
        }
    

    这个 .statsId(new Long(1)) 是个问题,因为它被认为是一个分离的实体,因为有一个id。 CascadeType.MERGE CascadeType.ALL 会很好用的。

    @Test
        public void testSavePeopleWithOneStats() {
    
            // Creates People entity
            People people = this.generatePeopleWithoutId();
            // Retrieve existing stats from StatsRepository
            Stats stats = this.statsRepository.findById(new Long(1)).get();
            // Add Stats to People
            people.addStats(stats);
            // Persist and retrieve
            People p = this.entityManager.persistFlushFind(people);
    
            assertThat(p.getPeopleStats().size()).isEqualTo(1);
        }
    

    我有一个 data-h2.sql 在单元测试开始时加载在统计数据中的脚本,这就是为什么我可以从 statsRepository

    我也更新了我的 poc in github .

    希望这对下一个来的人有帮助。