posts - 64,  comments - 9,  trackbacks - 0
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 cityId
        private int couter;

        public String loadCity() throws DataAccessException, BaseException{
        counter ++;
                return "city";
        }
}
然后,这里上面的类中的成员类MthCity的定义如下
public class MthCity implements java.io.Serializable {
       
        private Long cityId       
        private 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 -> Action
Jsp页面
        以下是一个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" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/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"
    "http://struts.apache.org/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/*.而且mothed={1},所以方法名由链接指定
也就是loadCity.do所指定的.loadCity方法.

        这个do方法后面是可以带参数的.所带的参数名,要是CityAction中定义的成员变量,包括成员类.例如,如果想提交后,CityAction中的cityId有值,链接可以这样写
http://localhost/admin/city/loadCity.do?cityId=9
这样,在loadCity方法中,如果你访问cityId,就可以发现cityId的值是9
System.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 cityId
        private 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"
"http://hibernate.sourceforge.net/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="java.lang.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="java.lang.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的出错.

posted on 2009-10-13 11:29 super_nini 阅读(644) 评论(0)  编辑  收藏

只有注册用户登录后才能发表评论。


网站导航:
 
<2009年10月>
27282930123
45678910
11121314151617
18192021222324
25262728293031
1234567

常用链接

留言簿

随笔档案

文章档案

相册

搜索

  •  

最新评论

阅读排行榜

评论排行榜