我认输了。我不明白,能有多难?
我有两个实体User
和Group
,具有多对多的关系。 Group
管理关系。所以在Group
我有:
@Entity
@Table(name = "GROUPS", catalog = "", schema = "GROUPADMIN")
public class Group {
...
@ManyToMany
@JoinTable(
name = "GROUP_USERS",
joinColumns = {@JoinColumn(name = "GROUP_ID")},
inverseJoinColumns = {@JoinColumn(name = "USER_ID")}
)
private Set<User> users;
然后User
我创建实体如下内容:
@Entity
@Table(name = "USERS", catalog = "", schema = "GROUPADMIN")
public class User {
...
@ManyToMany(mappedBy="users")
private Set<Group> groups;
然后在我的后备 bean(实际上是@Named("registry")
)中,我将对检索到的user
的引用存储为具有相同名称的属性。
然后我在我的 JSF 中皇家使用该支持豆
Hello <h:outputLabel value="#{registry.user.firstName}"/>
<h:panelGroup>
<h:dataTable value="#{registry.user.groups}" var="g">
<f:facet name="header">Properties List</f:facet>
<h:column>
<f:facet name="header">Group</f:facet>
<h:outputText value="#{g.id}"/>
</h:column>
</h:dataTable>
</h:panelGroup>
对于那些对表格感兴趣的人:
create table "GROUPADMIN".GROUPS
(
ID VARCHAR(15) not null primary key
);
create table "GROUPADMIN".USERS
(
ID VARCHAR(50) not null primary key,
PASSWORD VARCHAR(50),
FIRST_NAME VARCHAR(50),
LAST_NAME VARCHAR(50)
);
create table "GROUPADMIN".GROUP_USERS
(
GROUP_ID VARCHAR(15) not null,
USER_ID VARCHAR(50) not null,
primary key (GROUP_ID, USER_ID)
);
首先注意到的是组似乎是空的,不是空的,而是空的(通过调试等验证)。所以我知道默认情况下@ManyToMany
会使用 Lazy 绑定,所以我将其更改为 fetch = FetchType.EAGER
(这甚至无关紧要)。这样做之后,事情真的变得奇怪了......
此时EL开始抱怨id属性在"#{g.id}"
上不存在:
Caused by:
javax.el.PropertyNotFoundException - Property 'id' not found on type org.apache.openjpa.util.java$util$HashSet$proxy
at javax.el.BeanELResolver$BeanProperties.get(BeanELResolver.java:266)
为什么它试图在HashSet
代理上检索属性id
,而不是在Group
上?
所以我很好奇我实际上得到了什么 g
我从<h:dataTable>
获得的属性,我试图简单地使用 <h:outputText value="#{g}"/>
输出它......结果真的很有趣...
g
打印出来的不仅仅是Set<Group>
中的一个元素,而是集合本身,这与前面显示的异常相匹配......这更能说明 API 或集成中的错误?
所以基本上看起来有几件事被破坏了,尽管我怀疑这一切都与同一个问题有关。
请注意,我使用了基本的现成的TomEE,最新版本,没有自定义(也没有JPA的增强器)。
为什么它应该工作:JSR 338:JavaTM Persistence 2.1。
提交的错误:Jira 问题 OpenJPA-2546(请投票解决此问题)。
解决办法:将Set<T>
替换为List<T>
作为解决办法。