Seam新手实战(4):外键

来源:互联网 发布:淘宝网上有极度秒杀吗 编辑:程序博客网 时间:2024/04/30 04:40
一个图书管理系统中的两个对象: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>
点击“SelectbookType”页面就自动跳转到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联系我。。