iBatis的JpetStore示例中MVC机制实现的研究

JpetStore中的Action与普通Struts的Action处理方式不一样。遍历JpetStore的src文件夹,并无一个具体的Action,那么它是如何来完成普通Struts的Action工作了?
查看JpetStore的Struts.xml可以发现,它的Action只有一个,即“org.apache.stuts.beanaction.Beanaction”。通过Eclipse查看beanaction.jar的源代码,可以发现Beanaction继承与普通Action,即具备普通的action的功能。那么它无具体Action的奥妙在哪,继续研究BeanAction的代码,截取BeanAction的excute方法中核心部分代码如下:

/**/ /*遍历此方法的业务逻辑可知,*的优先级最高,如果是*,则不调用任何方法直接Forward,类似于ForwardAction*/
private static final StringNO_METHOD_CALL = " * " ;
…….
/**/ /*所有的FormBean都继承于BaseBean*/
BaseBeanbean
= (BaseBean)form;
ActionContext.initCurrentContext(request,response);
if (bean != null ) {
//ExplicitMethodMapping
/**//*下面是检查struts.xml配置中是否有parameter属性*/
Methodmethod
=null;
StringmethodName
=mapping.getParameter();
if(methodName!=null&&!NO_METHOD_CALL.equals(methodName)){
try{
/**//*通过反射,根据得到的方法名称取得方法的句柄*/
method
=bean.getClass().getMethod(methodName,null);
synchronized(bean){
/**//*下面是关键一句,调用basebean拥有的接口ActionInterceptor的实现DefaultActionInterceptor,来完成具体方法的调用*/
forward
=bean.getInterceptor().intercept(newActionInvoker(bean,method));
}

……..
/**//*无Parameter属性,检查path路径的最后一个/后的名称,即为调用的方法名*/
//PathBasedMethodMapping
if(method==null&&!NO_METHOD_CALL.equals(methodName)){
methodName
=mapping.getPath();
if(methodName.length()>1){
intslash=methodName.lastIndexOf("/")+1;
methodName
=methodName.substring(slash);
if(methodName.length()>0){
try{
method
=bean.getClass().getMethod(methodName,null);
synchronized(bean){
forward
=bean.getInterceptor().intercept(newActionInvoker(bean,method));
}

……..
/**//*根据调用方法返回的String,得到页面的转移路径*/
returnmapping.findForward(forward);

通过研究上面这段代码,我们可知,JpetStore中没有具体Action实现的关键原因即在于下面这几句

/**/ /*通过反射,根据得到的方法名称取得方法的句柄*/
method
= bean.getClass().getMethod(methodName, null );
synchronized (bean) {
/**//*下面是关键一句,调用basebean拥有的接口ActionInterceptor的实现DefaultActionInterceptor,来完成具体方法的调用*/
forward
=bean.getInterceptor().intercept(newActionInvoker(bean,method));
}

即将原来Action中的excute方法的实现转移到FormBean中,这样实现显得更为简捷,方便。研究ActionInvoke,它的核心代码如下:

public Stringinvoke() {
try{
return(String)method.invoke(bean,null);
}
catch(Exceptione){
thrownewBeanActionException("ErrorinvokingAction.Cause:"+e,e);
}

}

至此可知,它调用的是formbean中的函数。且从这段代码可知,formbean的这类特殊函数,此处称为action方法,要符合两个特征:1)无参数;2)返回值为string,此返回string即是Struts-config.xml的全局或局部的forward。
以上是整个beanaction的实现机制。个人感觉此种实现方法对于开发者而言已经类似于ASP.NET的.aspx与.cs开发模式了。下面是通过实例来说明一下BeanAction如何控制formbean的

Struts-config.xml的配置里有3种映射方式,来告诉BeanAction把控制转到哪个form bean对象的哪个方法来处理。
(1)parameter=”*’直接跳转;(2)Parameter中含具体的方法名;(3)Path中最后一个/后的方法名
以这个请求连接为例http://localhost/jpetstore4/shop/viewOrder.shtml
1. URL Pattern

< action path ="/shop/viewOrder" type ="com.ibatis.struts.BeanAction"
name
="orderBean" scope ="session"
validate
="false" >
< forward name ="success" path ="/order/ViewOrder.jsp" />
action>

此种方式表示,控制将被转发到"orderBean"这个form bean对象 的"viewOrder"方法(行为)来处理。方法名取"path"参数的以"/"分隔的最后一部分。
2. Method Parameter

< action path ="/shop/viewOrder" type ="com.ibatis.struts.BeanAction"
name
="orderBean" parameter ="viewOrder" scope ="session"
validate
="false" >
< forward name ="success" path ="/order/ViewOrder.jsp" />
action>

此种方式表示,控制将被转发到"orderBean"这个form bean对象的"viewOrder"方法(行为)来处理。配置中的"parameter"参数表示form bean类上的方法。"parameter"参数优先于"path"参数。
3. No Method call

< action path ="/shop/viewOrder" type ="com.ibatis.struts.BeanAction"
name
="orderBean" parameter ="*" scope ="session"
validate
="false" >
< forward name ="success" path ="/order/ViewOrder.jsp" />
action>

此种方式表示,form bean上没有任何方法被调用。如果存在"name"属性,则struts把表单参数等数据填充到form bean对象后,把控制转发到"success"。否则,如果name为空,则直接转发控制到"success"。
这就相当于struts内置的org.apache.struts.actions.ForwardAction的功能

< action path ="/shop/viewOrder" type ="org.apache.struts.actions.ForwardAction"
parameter
="/order/ViewOrder.jsp" scope ="session" validate ="false" >
action>

你可能感兴趣的:(mvc,bean,jsp,ibatis,struts)