一个图书管理系统中的两个对象:Book(书籍),BookType(书籍类型)。Book和BookType之间是多对一关系。
说到主从表的关联关系,自然而然地想起的一种实现方式就是选择框,比如在Book的编辑界面是使用一个类型的下拉选择框,选择一个类型,然后保存。于是就有下例代码:
<h:selectOneMenu id="type" value="#{bookHome.instance.bookType.typeId}">
<f:selectItems value="#{bookTypeList.typeSelectItems}" />
</h:selectOneMenu>
这段代码的作者(我)原本的想法是让这个下拉框与bookHome.instance里的
bookType.typeId帮定。比如当前的book类型id是1,修改后将book类型的id改为2,更新到数据库。但是很不幸。这段代码并不能执行预期的行为,或者说它还附加了其他行为。即Seam已经觉察到了在这个book中的类型的一个属性(主键值)已经改变了。于是,试图更新这个类型。但是JPA的规范中是不允许更改主键的,这就引起了一个错误。不信可以试下哦,呵呵。(以上描述我已经尽可能说清楚我的想法,但可能还是不怎么清楚,希望大家看不懂的说说哪里看不懂,我好改正)。我想说的是,不要直接帮定到外键,而是现帮定到一个临时变量,比如在bookHome中多写一个变量:
public class BookHome extends EntityHome<Book>{
private Long typeId;
//..getter setter
}
然后将下拉框邦定到这个变量上,比如:
<h:selectOneMenu id="type" value="#{bookHome.typeId}">
<f:selectItems value="#{bookTypeList.typeSelectItems}" />
</h:selectOneMenu>
。然后在重载的persist或者update方法中写上:
BookType newType = getEntityManager().find(BookType.class, typeId);
instance.setType(newType);
return super.persist();
就完成了。我叫它“转移邦定”,呵呵
Seam的解决方案:
其实Seam有另一种解决方案。比如如果你是自动生成代码的方式,在BookEdit.xhtml中就会看到这样的代码:
<div class="association" id="bookTypeParent">
<h:outputText value="There is no bookType associated with this book."
rendered="#{bookHome.instance.bookType == null}"/>
<rich:dataTable var="bookType"
value="#{bookHome.instance.bookType}"
rendered="#{bookHome.instance.bookType != null}"
rowClasses="rvgRowOne,rvgRowTwo"
id="bookTypeTable">
<h:column>
<f:facet name="header">typeId</f:facet>
#{bookType.typeId}
</h:column>
<h:column>
<f:facet name="header">bookType typeId</f:facet>
#{bookType.bookType.typeId}
</h:column>
<h:column>
<f:facet name="header">typeName</f:facet>
#{bookType.typeName}
</h:column>
<h:column>
<f:facet name="header">action</f:facet>
<s:link view="/BookType.xhtml"
id="viewbookType"
value="View"
propagation="none">
<f:param name="bookTypeTypeId"
value="#{bookType.typeId}"/>
</s:link>
</h:column>
</rich:dataTable>
<div class="actionButtons">
<s:button value="Select bookType"
view="/BookTypeList.xhtml">
<f:param name="from" value="BookEdit"/>
</s:button>
</div>
</div>
点击“Select bookType”页面就自动跳转到BookTypeList页面,列出所有的类型,每个类型后面都有一个select连接,点击这个连接就可选中这个类型。然后回到BookEdit.xhtml。很Seam很强大吧,呵呵。其实为何这样能完成一个选择都是在BookEdit.page.xml里配置的。配置大概如下:
<begin-conversation join="true"/>
<action execute="#{bookHome.wire}"/>
<param name="bookFrom"/>
<param name="bookBookId" value="#{bookHome.bookBookId}"/>
<param name="bookTypeFrom"/>
<param name="bookTypeTypeId" value="#{bookTypeHome.bookTypeTypeId}"/>
<begin-conversation join="true"/> :开始一个conversation(我暂称之为“页面流”),如果已存在,就加入。而不重新创建。
<action execute="#{bookHome.wire}"/> :一最重要的这个。如果没有执行这个方法这段跳转就没有任何效果了。,先来看下这个方法是怎么写的吧:
public void wire() {
getInstance();//获取instance,放在这里是为了加在instance
BookType bookType = bookTypeHome.getDefinedInstance();//获取类型。
if (bookType != null) {//如果选择的类型不为null
getInstance().setBookType(bookType);//设置书籍类型
}
}
那bookTypeHome从哪来的呢?天上掉下的?呵呵,当然不是。就在BookHome的上部分:
@In(create = true)
BookTypeHome bookTypeHome;
前面我说过了。这个@In就是拿来做双向注入的。
bookTypeHome是要注入的组件名称。
其他的都是参数了,没啥好解释的。
但为什么在BookTypeList页面点select,怎么就会自动跳转到BookEdit.xhtml呢?奥秘就在这段代码里(BookTypeList.xhtml):
<s:link view="/#{empty from ? 'BookType' : from}.xhtml"
value="Select"
id="bookType">
<f:param name="bookTypeTypeId"
value="#{bookType.typeId}"/>
</s:link>
从BookEdit里传来一个from。
<s:button value="Select bookType"
view="/BookTypeList.xhtml">
<f:param name="from" value="BookEdit"/>
</s:button>
就告诉了BookTypeList,是从BookEdit里来的,点Select的时候就不要去其他地方了。直接回去。
好了,今天到这,困了,上面讲的不明白的欢迎email或qq联系我。。
posted on 2008-12-24 23:02
phyeas 阅读(1069)
评论(4) 编辑 收藏 所属分类:
Seam项目实战