最初想写zooma的原因是因为自己感到对经常性的重复劳动的厌烦,诸如一些po和vo的转换,针对不同entity的dao和service的实现,以及基本的action中crud操作,即使对dao和service以及action做了泛型的封装,还是要写不少的重复代码,前台的页面也是要来回的粘贴复制,于是我就有了zooma的原始模型:
就是通过hibernate的mapping文件自动的生成对entity的包括po,vo,dao,service,action,actionform和前台的页面,一次性生成对该实体的包括c、s两端的基本操作代码,当然包括spring和struts的配置文件的更新,使得我可以把大部分的精力放在对于特定实体的特定逻辑的设计上。
到zooma0.3版本,以上的模型已经全部实现,使用zooma为基础的山西某煤矿生产调度信息系统也同时完成收尾工作,在开发过程中遇到了zooma的先天缺陷,在后面会给大家介绍,希望使用它的人能够避免犯同样的错误。下面我来分析一下zooma的优点和缺点:
首先我对后台的代码做了优化,使用泛型对dao,service和action做了基本操作的封装,这样可以看到dao实现的代码如下:
public class ZsxResumeDAO extends BabyDao <ZsxResume,Integer>{
}
service实现的代码如下:
(1)oracle下的代码
public class ZsxResumeService extends BabyService<ZsxResumeDAO,ZsxResume,ZsxResumeVO>{
}
(2)mysql下的代码
public class ZsxResumeService extends BabyService<ZsxResumeDAO,ZsxResume,ZsxResumeVO>{
public boolean delete(String[] ids){
try {
for(int i=0;i<=ids.length;i++){
this.getDao().delete(Integer.parseInt(ids[i]));
}
return true;
} catch (Exception e) {
return false;
}
}
public ZsxResumeVO findById(String id) {
CoypUtil.copyProperties(this.getVo(),this.getDao().findById(Integer.parseInt(id)));
return this.getVo();
}
}
action实现的代码:
public class ZsxResumeAllAction extends BabyActionAdvan<ZsxResumeService,ZsxResume,ZsxResumeVO,ZsxResumeForm>{
/**
* Struts execute
*/
public ActionForward execute(ActionMapping mapping, ActionForm form,
HttpServletRequest request, HttpServletResponse response) {
String action=request.getParameter("action");
String actionForward="404Failed";
if ("insert".equals(action)) {
/*insert表示插入,由xxx_add.jsp跳入,跳转xxx_index.jsp*/
if (insert(form))actionForward="zsxResume_insert";
}else if("update".equals(action)) {
/*update表示修改,由xxx_update.jsp跳入,跳转xxx_index.jsp*/
if(update(form))actionForward="zsxResume_update";//xxx_index.jsp
}else if("show".equals(action)) {
/*show表示修改显示,由xxx_index.jsp或xxx_query.jsp跳入,跳转xxx_update.jsp*/
String id=request.getParameter("id");
vo= (ZsxResumeVO)(service).findById(id);
Vector<ZsxResumeVO> v=new Vector<ZsxResumeVO>();
v.add(vo);
request.setAttribute("result",v );
actionForward="zsxResume_show";
}else if("delete".equals(action)) {
/*delete表示删除,由xxx_index.jsp或xxx_query.jsp跳入,跳转xxx_index.jsp或xxx_query.jsp*/
String[] ids=request.getParameterValues("id");
if (!delete(ids))actionForward="zsxResume_delete";
}else{
/*query,由xxx_index.jsp或xxx_query.jsp跳入*/
Page p=query(form, request, "zsxResume");/*第三个参数需要相应修改,值为当前action的path名称*/
request.setAttribute("result",p.getItems());
request.setAttribute("page",p);
/*判断是主页跳入,还是查询页跳入*/
if ("index".equals(this.form)){
//主页跳入,应该跳回主页
actionForward = "zsxResume_index";//xxx_index.jsp
}else if("query".equals(this.form)){
//查询页跳入,应该跳回查询页
actionForward = "zsxResume_query";//xxx_query.jsp
}
}
return mapping.findForward(actionForward);
}
/**
* overrided begin
*/
/**
* ActionForm to VO
*/
public void copyForm(ZsxResumeForm form) {
ZsxResumeForm f = form;
if(f.getId()!=null) {
if(!f.getId().equals(""))vo.setId(new Integer(f.getId()));
}
if(f.getAge()!=null) {
if(!f.getAge().equals(""))vo.setAge(new Integer(f.getAge()));
}
if(f.getSalary()!=null) {
if(!f.getSalary().equals(""))vo.setSalary(new Float(f.getSalary()));
}
if(f.getBrithday()!=null) {
if(!f.getBrithday().equals(""))vo.setBrithday(DateUtil.stringToDate(f.getBrithday()));
}
vo.setName(f.getName());
vo.setCity(f.getCity());
}
/**
* query rules
*/
public DetachedCriteria getDetachedCriteria(ZsxResume po){
//建立查询规则
DetachedCriteria dc=DetachedCriteria.forClass(ZsxResume.class);
if(po.getName()!=null&&!po.getName().equals("")){
dc.add(Restrictions.like("name",po.getName(),MatchMode.ANYWHERE));
}
if(po.getCity()!=null&&!po.getCity().equals("")){
dc.add(Restrictions.like("city",po.getCity(),MatchMode.ANYWHERE));
}
if(po.getId()!=null&&!po.getId().equals("")){
dc.add(Restrictions.eq("id",new Integer(po.getId())));
}
if(po.getAge()!=null&&!po.getAge().equals("")){
dc.add(Restrictions.eq("age",new Integer(po.getAge())));
}
if(po.getSalary()!=null&&!po.getSalary().equals("")){
dc.add(Restrictions.eq("salary",new Float(po.getSalary())));
}
if(po.getBrithday()!=null&&!po.getBrithday().equals("")){
dc.add(Restrictions.eq("brithday",po.getBrithday()));
}
dc.addOrder(Order.desc("id"));
return dc;
}
}
这些代码相对在量上有了明显的压缩,也更加的适合扩展,比如我希望在dao中添加一个按照name来查询的方法,直接在dao里添加如下方法就可以了:
public List findByName(String name){
return this.getHibernateTemplate().find("from ZsxResume",name);
}
再比如我想使用其它实体的操作,只要在这个实体中用geter和seter注入就可以使用,当然相应的要在spring的配置文件中加入参数信息。
在action中只实现了单独实体的crud操作,想实现批量操作,可以自己重写insert(),delete(),update()方法,同样也可以通过ioc使用其他实体的crud操作
在查询方面使用hibernate的QBC进行动态混合查询,查询规则在action中的getDetachedCriteria方法中定义,同时将分页封装了进去,可以自己修改org.zooma.base.Page类的toString()方法,来改变翻页显示条。
页面中尽量将java代码最少化,使得代码看起来更加易懂,所以使用了struts的标签,
这里有个缺陷就是页面上按钮使用的js跳转,所以大家可以看到用我提供的模板生成的jsp文件需要使用js,css和images三个文件夹中的文件。
在WEB-INF\templete下存放的是xml和xsl模板,zooma会先将mapping文件里的信息保存在xml\bean.xml中,然后使用它和xsl文件夹下的模板进行文件生成,一次前台页面的生成样式是完全可以自己定义的,只要按照bean.xml中的数据进行编写就可以了,当然名字还是要叫那个名字。
zooma的缺陷:
在项目中我是这样使用zooma的,单表直接只用zooma生成就ok了,多表关联的时候,先建立视图,然后对视图做生成——这里对表和视图有一定的限制,就是一定要以id为主键,并且id在oracle中使用number、在mysql中使用integer,否则会出错误,这个应该算是个缺陷吧——生成的视图用来显示关联后的实体数据,对视图修改的时候,却修改要被修改的被关联的单表,如果说对于比较简单的逻辑关系这样操作还是没有问题的,但是我们在实际项目中遇到的比较复杂的逻辑,使得实现的十分复杂,所以建议对多表关联使用zooma对单边生成后自己修改mapping文件和po、vo,使用hibernate的many-to-many进行操作,这是zooma得最大缺陷,它只能为开发者在比较简单的实体逻辑(一般我会在这样的逻辑中使用zooma:单表,many-to-one,one-to-many)下生成应用,使用它来做一些小型项目我绝得还是会很有益处的。
还有就是zooma是基于struts1.2的,这令很多网友感到失望,我在演示视频中使用myeclipse进行项目搭建,有让很多人感到绝望,实际上完全可以除了基本架构的搭建,其他操作完全可以不使用myeclipse,但是需要自己手动写mapping文件和po。
在zooma的下个版本中希望实现:
1、ant的项目导入,摆脱Myeclipse
2、升级为Struts2.0,有可能的话会添加ibates支持
分享到:
相关推荐
其次,【资源管理】的讨论包括了客户订单需求分析和设备稼动率。订单需求分析揭示了内外观件的周趋势,例如在WK11和WK12期间,有特定产品逐步接手交货。设备稼动率分析显示,虽然大部分制程如冲压、抛光、拉丝和喷砂...
### 产品特点与功能 #### 1. 多种效果预设 ZOOM A3内置了丰富的音效预设,覆盖了从传统到现代的各种风格,包括但不限于: - **失真(Distortion)**:模拟各种放大器的失真效果,增加声音的厚度和冲击力。 - **过载...
- "main.dfm"、"line.dfm"、"flash.dfm"、"zoomx.dfm"、"Zooma.dfm":这些都是表单文件,定义了应用程序的界面布局和组件。例如,"flash.dfm"可能与SWF文件的显示有关,而"main.dfm"可能代表主界面。 - "keyan125....
AutoCAD软件是一款广泛应用于计算机辅助设计(CAD)领域的软件,它允许用户在计算机上创建精确的二维和三维设计。根据给定文件的部分内容,我们可以推断出一些关于AutoCAD软件的基本设置知识点。 首先,文件中提到...
KWDB 是一款面向 AIoT 场景的分布式多模数据库产品,支持在同一实例同时建立时序库和关系库并融合处理多模数据,具备千万级设备接入、百万级数据秒级写入、亿级数据秒级读取等时序数据高效处理能力,具有稳定安全、高可用、易运维等特点。
yolo系列算法目标检测数据集,包含标签,可以直接训练模型和验证测试,数据集已经划分好,适用yolov5,yolov8,yolov9,yolov7,yolov10,yolo11算法; 包含两种标签格:yolo格式(txt文件)和voc格式(xml文件),分别保存在两个文件夹中; yolo格式:<class> <x_center> <y_center> <width> <height>, 其中: <class> 是目标的类别索引(从0开始)。 <x_center> 和 <y_center> 是目标框中心点的x和y坐标,这些坐标是相对于图像宽度和高度的比例值,范围在0到1之间。 <width> 和 <height> 是目标框的宽度和高度,也是相对于图像宽度和高度的比例值
那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据
做4.3这一节的时候的maple文件,仅供参考
stm32的实时时钟使用代码
基于go语言,使用gocv和socket实现摄像头视频传输项
jsoncpp不能正常解析,以及全角字符的问题,可以直接编辑使用.zip
在我们日常使用电脑的过程中,经常会遇到需要在不同网络环境下切换 IP 地址的情况。手动设置 IP 地址不仅繁琐,还容易出错。今天,我要向大家推荐一款超实用的网络管理工具 ——IP Switcher。 一、软件简介: IP Switcher 是一款功能强大的网络配置切换软件,它可以帮助用户在不同的网络环境下快速切换 IP 地址、子网掩码、网关、DNS 等网络设置,提高工作效率。 二、软件特点: 快速切换 IP Switcher 可以在几秒钟内完成网络配置的切换,无需手动设置 IP 地址、子网掩码、网关、DNS 等参数,大大节省了时间。 多种配置方案 用户可以根据不同的网络环境创建多个网络配置方案,每个方案可以设置不同的 IP 地址、子网掩码、网关、DNS 等参数。在需要切换网络环境时,只需选择相应的配置方案即可。 自动切换 IP Switcher 支持自动切换网络配置方案,可以根据用户设置的条件自动切换到相应的网络配置方案。例如,用户可以设置在连接到特定的无线网络时自动切换到相应的网络配置方案。 简单易用 IP Switcher 的界面简洁直观,操作非常方便。用户只需几个简单的步骤
tornado创建的一个web项目,实现了cookie,session,连接mysql和redis数据库,对主handler进行抽取,模拟登陆,图形化验证等一些功能业务_tornado_project.zip
mtk计算屏帧数的表格
fenlei20241031
那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据
爱心代码
那些年,与你同分同位次的同学都去了哪里?全国各大学在四川2020-2024年各专业最低录取分数及录取位次数据,高考志愿必备参考数据