dynafuse 1.0 在sourceforge上发布!

发表于:2007-07-01来源:作者:点击数: 标签:
1. 概述 DynaFuse类似Appfuse,是一套整合目前主流J2EE技术的新构架,使用到的技术有struts,springframework,hibernate,groovy等,提出了Evertything be Dynamic的理念,从DAO到WEB层提供了全新的动态建模,动态编程的方案。2. 运行例子 下载并安装 mysql
1. 概述   DynaFuse类似Appfuse,是一套整合目前主流J2EE技术的新构架,使用到的技术有struts,springframework,hibernate,groovy等,提出了Evertything be Dynamic的理念,从DAO到WEB层提供了全新的动态建模,动态编程的方案。2. 运行例子
下载并安装mysql-4.0,建立数据库dynafuse,建立用户username/password为admin/admin。

下载并解压dynafuse-xx.zip,在mysql中执行data/create-tables.sql中的语句。

下载jakarta-tomcat-5.0.28,将dynafuse\lib\mysql-connector-java-3.0.14-production\mysql-connector-java-3.0.14-production-bin.jar和dynafuse\lib\jta-1.0\jta.jar拷贝到jakarta-tomcat-5.0.28\common\lib下,将dynafuse\dynafuse.xml拷贝到jakarta-tomcat-5.0.28\conf\Catalina\localhost下,

下载dynafuse.war拷贝到jakarta-tomcat-5.0.28\webapps下。
启动tomcat,打开浏览器http://localhost:8080/dynafuse进入例子程序。 3. 包结构说明
下载并解压dynafuse包,会得到如下的目录结构:



build:编译后文件

config:配置文件

data:试验数据

dist:打包后的发布文件

docs:文档

lib:使用到的jar文件

metadata:xdoclet需要的配置文件片断

script:groovy脚本

src:源码

test:测试源码

web:例子程序的jsp等文件
4. 构架说明 dynafuse最主要的特征是提倡Everything be Dynamic,从dao到service到web三个层面,每个层面都提供了对groovy新型server端srcipt语言的整合,使得web应用中基本所有的逻辑代码都能够以动态的形式编写,维护,构架分为三层:   DAO层 - 这个层面中,dynafuse提供了两个interface:DAO 和 DynaDAO ,DAO是一个通用的数据访问(增、删、改、查)接口,DynaDAO在DAO的基础上提供了对DAO层面脚本的调用功能。这两个interface目前只提供了hibernate的实现。 DAO和DynaDAO分别作为springframewok中的bean在IOC容器中进行配置,详细请参考config\applicationContext-hibernate.xml。   Service层 - 这个层面中,dynafuse提供了两个interface:Manager 和 DynaManager,完成对DAO的封装并提供对service层面中业务逻辑脚本的调用功能,调用service层面脚本有两个方法,一是在事务环境中执行脚本(invokeScriptInTransaction),一个是不需要事务(invokeScript)。Manager和DynaManager分别作为springframewok中的bean在IOC容器中进行配置,详细请参考config\applicationContext-service.xml。   Web层 - 这个层面中,dynafuse提供了DynaAction,完成对web层面脚本的调用。

5. 构架使用
下面的文字将描述如何利用dynafuse进行web应用程序的开发

1.构建hibernate的动态模型

   hibernate在正式发布的3.0版本中,提供了动态模型,关于动态模型,可以查阅hibernate的相关资料,下面给出的是dynafuse示例程序的动态模型,文件参见dynafuse\model\Example.hbm.xml

<?xml version="1.0"?>        <!DOCTYPE hibernate-mapping PUBLIC        "-//Hibernate/Hibernate Mapping DTD 3.0//EN"        "http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">       <hibernate-mapping default-lazy="false">       <class entity-name="Dept" table="t_dept">           <id                name="id"                column="dept_id"                type="java.lang.Integer"                unsaved-value="null"            >                <generator class="native">                </generator>           </id>           <property                name="name"                type="java.lang.String"                update="true"                insert="true"                column="dept_name"                not-null="true"                unique="false"             />             <property                name="description"                type="java.lang.String"                update="true"                insert="true"                column="dept_description"                not-null="false"                unique="false"              />              <bag                 name="persons"                 lazy="true"                 inverse="true"                 cascade="delete"               >                 <key                      column="deptid"                  >                  </key>                  <one-to-many                      class="Person"                  />              </bag>        </class>        <class entity-name="Person" table="t_person">                 <id                     name="id"                     column="person_id"                     type="java.lang.Integer"                     unsaved-value="null"                 >                   <generator class="native">                   </generator>                </id>                <property                     name="name"                     type="java.lang.String"                     update="true"                     insert="true"                     column="person_name"                     not-null="true"                     unique="false"                 />                 <property                      name="age"                      type="java.lang.Integer"                      update="true"                      insert="true"                      column="person_age"                      not-null="true"                      unique="false"                  />                  <property                       name="description"                       type="java.lang.String"                       update="true"                       insert="true"                       column="person_description"                       not-null="false"                       unique="false"                    />                    <many-to-one                          name="dept"                          class="Dept"                          cascade="none"                          update="true"                          insert="true"                          column="deptId"                     />        </class>

       </hibernate-mapping>

这里没有静态的POJO持久类,所有的持久数据都通过Map结构来存放。

2.DAO层面的脚本,如果DAO接口中提供的方法不能满足需要,那么可以在DAO层面加入脚本,然后在service层通过DynaDAO完成对DAO脚本的调用,下面是示例程序中的dynafuse cript\PersonDAO.groovy

import org.hibernate.Session;       import org.hibernate.Query;       public class PersonDAO {

             public Session session;             public Object getAvgAgeOfDept(Integer deptId) {                 String hsql = "select avg(t.age) from Person t where t.dept.id=:deptId";                 Query query = session.createQuery(hsql);                 query.setInteger("deptId", deptId.intValue());                 return query.uniqueResult();              }         }

这里需要说明的是session这个成员变量,它是必须要在DAO层面的脚本中申明的,在dao层面的脚本被执行的时候,dynafuse将会自动对这个session赋值,在脚本中,你不用关心session是怎么得到,你只管放心使用。

3.service层面的脚本,在实际的应用中,业务逻辑是千变万化的,所以service层面加入脚本支持是非常有用的,service层的脚本将在web层通过DynaManager被调用,下面是示例程序中的dynafuse cript\DeptManager.groovy

import org.robusta.dynafuse.dao.DynaDAO;       import java.util.List;       public class DeptManager {

             public DynaDAO dynaDao;             public List getDepts() {                 return dynaDao.getObjects("Dept", null, null, null);             }             public Object getAvgAgeOfDept(Integer deptId) {                 return dynaDao.invokeScript("PersonDAO.groovy", "getAvgAgeOfDept", new Object[]{deptId});             }        }

这里需要说明的是dynaDao这个类成员变量,和dao层面脚本中的session一样,是在service层面的脚本中执行的时候,由dynafuse自动赋值,你只管放心使用它。

还需要注意的是,在service层面的脚本中可以利用dynaDao去调用DAO层面的脚本,如上面的getAvgAgeOfDept。

4.构建web层的动态模型,struts提供了DynaValidatorActionForm这种构造动态模型的机制,请参考示例中的dynafuse\metadata\web truts-forms.xml。

5.web层面的脚本,跟业务逻辑一样,页面逻辑也是千变万化的,所以web层的脚本非常有用,下面是示例程序中的dynafuse cript\DeptAction.groovy

public class DeptAction extends BaseAction {

     ....

     public DynaManager dynaManager;

     ....

     public ActionForward editDept(ActionMapping mapping, ActionForm form,                                                         HttpServletRequest request, HttpServletResponse response) {                 if (log.isDebugEnabled()) {                    log.debug("Entering @#editDept@# method");                 }                 String deptId = request.getParameter("deptId");                 if (deptId != null && !"".equals(deptId)) {                     Map data = dynaManager.getObject("Dept", new Integer(deptId));                     Long avgAge = (Long) dynaManager.invokeScript("DeptManager.groovy", "getAvgAgeOfDept", new Object[]{new Integer(deptId)});                     DynaValidatorActionForm deptForm = (DynaValidatorActionForm) form;                     deptForm.set("id", data.get("id"));                     deptForm.set("name", data.get("name"));                     deptForm.set("description", data.get("description"));                     deptForm.set("avgAge", avgAge);                     updateFormBean(mapping, request, deptForm);                  }                  return mapping.findForward("edit");             }

      ....

和servcie和dao层的脚本一样,web层脚本中有一个dynaManager成员变量。

6.将web层的脚本配置到action中,可以参考dynafuse\metadata\web truts-actions.xml

....

  <action              path="/depts"              type="org.robusta.dynafuse.web.action.DynaAction"              parameter="DeptAction.groovy/listDepts"              unknown="false"              validate="false"          >              <forward                   name="list"                   path="/WEB-INF/pages/deptList.jsp"                   redirect="false"               />          </action>

   ....

需要注意的是这里的parameter,指定了web层某个脚本中的某个method。

原文转自:http://www.ltesting.net