当前位置:文档之家› Struts2+Hibernate开发笔记

Struts2+Hibernate开发笔记

Struts2 + Hibernate开发笔记(一)由于开发任务紧张,因为这里的开发笔记,仅用于记录遇到的几个struts2和hibernate结合开发的现象.不对其做分析.1. 在使用struts2时,页面和action之间的传值这是struts2和struts1最大的区别.Struts2中,action和jsp页面之间的信息交互,是通过action中定义的成员变量来实现的.例如,我在一个名为EstateAction的类中有如下定义public class CityAction extends BaseAction {private MthCity mthCity ;private String cityName;private Long cityIdprivate int couter;public String loadCity() throws DataAccessException, BaseException{counter ++;return "city";}}然后,这里上面的类中的成员类MthCity的定义如下public class MthCity implements java.io.Serializable {private Long cityIdprivate String cityName;public MthCity() {public Long getCityId() {return this.cityId;}public void setCityId(Long cityId) {this.cityId = cityId;public String getCityName() {return this.cityName;}public void setCityName(String cityName) {this.cityName = cityName;}}这是一个Hibatenate使用的数据对象POJO类.有了这两个类后,我们来看看Struts2的Action和JSP页面之间是如何交互的一. JSP -> ActionJsp页面以下是一个jsp页面submit.jsp.这个页面只有一个功能,就是向struts提交申请<%@ page language="java" contentType="text/html; charset=gbk"%><%@ include file="/common/taglibs.jsp"%><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN""/TR/xhtml1/DTD/xhtml1-transitional.dtd"><html xmlns="/1999/xhtml"><head><meta http-equiv="Content-Type" content="text/html; charset=gbk" /><title>提交</title></head><script>function go (){window.location ="${pageContext.request.contextPath}/admin/city/loadCity.do”;}</script><body><form name=”myform” ><input type="button" name="cityupdate" id="cityupdate" value="编辑" onclick="javascript:go();"/> <input type="hidden" name="mthCity.cityName" id=" mthCity " value="广州" /></form></body></html>大家可以看到,这个页面只有一个按钮,将页面提交到struts的一个action中,这是为什么呢.我们先看一段struts2的配置文件admin-action.xml<?xml version="1.0" encoding="UTF-8" ?><!DOCTYPE struts PUBLIC"-//Apache Software Foundation//DTD Struts Configuration 2.0//EN""/dtds/struts-2.0.dtd"><struts><package name="admin" namespace="/admin" extends="struts-default"><action name="city/*" method="{1}"class="com.mytophome.admin.representation.struts.action.CityAction"><result name="city">/admin/city.jsp</result><result name="city_update">/admin/city_update.jsp</result></action></package></struts>这是一个struts2的典型配置文件.上面有几处要注意的首先是namespace = “/admin” 这是一个struts模块名,可以不写,但如果写了,能比较方便的将struts2的action按配置来分模块.(何谓分模块呢?struts2有个特性,是action定义不需要像struts1一样写在同一个struts.xml文件中.而是可以使用include的形式.例如我使用的项目的struts.xml文件就是这样写的:<struts><include file="struts-action/admin-action.xml"/><include file="struts-action/agent-action.xml"/></struts>这样include了一系统的xml配置,而上面的admin-action.xml文件就是其中一段,因此将这一段中涉及的action类设定为一个模块,就定namespace = “/admin”)其次<action name="city/*" method="{1}"这一句配置的意思,就是,当用户提交一个符合struts2格式的申请时(所有包含.do形式的http链接)例如http://localhost/admin/city/loadCity.do其中包含了/city/那么在配置文件中,只要定义action name=”city/*”,那么所有包含有/city/的do,都会提交到action定义的类中来,也就是类om.mytophome.admin.representation.struts.action.CityAction中,那么提交到这个类的哪个方法中呢? 因为选择的是city/*.而且method={1},所以方法名由链接指定也就是loadCity.do所指定的.loadCity方法.λ这个do方法后面是可以带参数的.所带的参数名,要是CityAction中定义的成员变量,包括成员类.例如,如果想提交后,CityAction中的cityId有值,链接可以这样写http://localhost/admin/city/loadCity.do?cityId=9这样,在loadCity方法中,如果你访问cityId,就可以发现cityId的值是9System.out.println(Long.toString(cityId));但这里有一个条件,就是CityAction中,必须要有cityId变量的getter/setter方法(这两个方法可以用MyEclipse自动生成)public Long getCityId() {return cityId;}public void setCityId(Long cityId) {this.cityId = cityId;}如果要给CityAction中的MthCity类的这样才能在jsp页面提交时,由struts为cityId赋值.(当然,getter方法就方便当action返回到jsp页面时,cityId的值能在页面取到.)λ如果要为action中的类成员变量赋值也是可以的例如http://localhost/admin/city/load ... mp;mthCity.cityId=8这条链接提交后,会和上面一样调用CityAction的loadCity方法,但这里,action的成员mthCity会被创建实例,并为mthCity的一个属性cityId赋值,也就是说,在action中,大家可以通过mthCity.getCityId()来获得这个属性的值.当然,一定要在action设置mthCity的getter setter,jsp上的内容才能传到action中public MthCity getMthCity()return mthCity;}public void setMthCity(MthCity city) {this. mthCity = city;}从JSP提交内容到Action,还有一种方法,就是将参数内容放在表单form的元素中 <input type="hidden" name="mthCity.cityName" id=" mthCity " value="广州" />这样,当用以下语句表单提交的时候doucment. myform.submit();就能在Action中创建一个mthCity实例,并为mthCity.cityName设置值为:广州.原因是在页面的表单元素中,设置了name= mthCity.cityName,而action中刚好有成员类叫mthCity,而这个类中刚好有属性叫cityName.,就是通过这样的方法,能将表单的内容,提交到Action中.Struts2 + Hibernate开发笔记(二)二. Action -> JSP当要从Action中执行的loadCity方法,要返回到jsp页面时,要在页面上指定一个return的页面.我们在admin-action.xml配置中可以看到一句<result name="city">/admin/city.jsp</result>而在loadCity方法中有这样一句return "city";这样,当loadCity执行完后,就会返回到http://localhost/admin/city.jsp页面.由于在action中,cityId和mthCity.cityId是被前一个提交过来的jsp页面赋值过,所以当程序执行转到city.jsp页面的时候,这几个值是能被使用的.我们使用jstl来获得这些值${cityId} , ${mthCity.citId} (前提是只要在action中,设定了cityId和mthCity类的getter/setter.)2. 当action中的类,传到受hibernate管理的命名空间的类中时这个问题比较特殊.我的工程中,建立了一个类cityService.这个类是进行数据库操作的,也就是和hibernate打交道的.而这个类在hibernate的设置中设置为被一个hibernate Session管理的范围.这个cityService中有一个方法,用于更新city的信息的public void updateCity(MthCity city) throws DataAccessException,BaseException{MthCity icity = this.getCityById( city.getCityId());icity.setCityName(city.getCityName());baseHibernateDAO.update(icity);//这个baseHibernateDAO只是一个封封装了hibernate API的包,网上常见.}大家可以看到,方法中并没有直接update从参数传进来的city,而是新建了一个icity,装了city中的信息再进行update.这是因为,cityService这个类被hibernate管理,所以在这个类中创建的内容,才能被更新.所以我们必须使用一个新的MthCity类实例来装着外面传进来的内容,才能更新.否则就会出现类似have the same id object in the session的错误,也就是说session中有其它相同id的对象的错.当然,有另一个处理办法,就是使用baseHibernateDAO.merge来更新内容,而还是用update这里就可以写成public void updateCity(MthCity city) throws DataAccessException,BaseException{baseHibernateDAO.merge(icity);}3. Hibernate中的version类型成员在我的项目中,city有一个属性是timestamp,对应的是mth_city这个表,这个表通过hibernate的映射,映射成一个POJO对象public class MthCity implements java.io.Serializable {private Long cityIdprivate Date timestamp;private String cityName;public MthCity() {}public Long getCityId() {return this.cityId;}public void setCityId(Long cityId) {this.cityId = cityId;}public Date getTimestamp() {return this.timestamp;}public void setTimestamp(Date timestamp) {this.timestamp = timestamp;}public String getCityName() {return this.cityName;}public void setCityName(String cityName) {this.cityName = cityName;}大家可以看到,类中有一个属性是timestamp,定义为Date类型.这个类型是用于记录数据库操作的日期的,数据库中的对应字段也叫timestamp.看一下hibernate的映射配置<?xml version="1.0" encoding="utf-8"?><!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN""/hibernate-mapping-3.0.dtd"><!--Mapping file autogenerated by MyEclipse Persistence Tools--><hibernate-mapping><class name="com.mytophome.admin.domain.MthCity" table="MTH_CITY"dynamic-update="true"><id name="cityId" type="ng.Long"><column name="CITY_ID" precision="10" scale="0" /><generator class="sequence" /></id><version name="timestamp" type="java.util.Date"><column name="TIMESTAMP" length="7" /></version><property name="cityName" type="ng.String"><column name="CITY_NAME" length="80" /></property></class></hibernate-mapping>大家可以看到,cityid对应数据库表中的CITY_ID字段,其值由oracle sequence生成而timestamp属性,对应的是数据库的TIMESTAMP字段,并且这个属性在mapping中定义为<version>属性这样,就会出现两点要注意的地方λ数据库中的这个timestamp的字段一定要有值,并且是日期值,否则当hibernate更新这个字段没有值的那条记录时,会出现如下错误org.hibernate.StaleObjectStateException: Row was updated or deleted by another transaction (or unsaved-value mapping was incorrect): [com.mytophome.admin.domain.MthCity#1] λ在更新MthCity中的值到数据库中,也就是更新一条记录时,一定不设调用方法人工设置timestamp属性的值,也就是下面的语句不能出现mthCity.setTimeStamp(new Date());否则也会引起hibernate的出错.。

相关主题