總結JavaWeb開發中SSH框架開發問題


在做JavaWeb的SSH框架開發的時候,遇到過很多的細節問題,這里大概記錄下

我使用的IDE是Eclipse(老版本)三大框架:Spring4、Struts2、Hibernate5

 

1.web.xml的配置

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://java.sun.com/xml/ns/javaee"
    xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"
    id="WebApp_ID" version="2.5">
    <display-name>Blog</display-name>
    <session-config>
        <session-timeout>60</session-timeout>
    </session-config>
    <!-- 讓spring隨web啟動而創建的監聽器 -->
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>
    <!-- 配置spring配置文件位置參數 -->
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:applicationContext.xml</param-value>
    </context-param>
    <!-- 擴大session作用范圍 -->
    <filter>
        <filter-name>openSessionInView</filter-name>
        <filter-class>org.springframework.orm.hibernate5.support.OpenSessionInViewFilter</filter-class>
    </filter>
    <!-- struts2核心過濾器 -->
    <filter>
        <filter-name>struts2</filter-name>
        <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>openSessionInView</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    <filter-mapping>
        <filter-name>struts2</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>

    <welcome-file-list>
        <welcome-file>index.html</welcome-file>
        <welcome-file>index.htm</welcome-file>
        <welcome-file>index.jsp</welcome-file>
        <welcome-file>default.html</welcome-file>
        <welcome-file>default.htm</welcome-file>
        <welcome-file>default.jsp</welcome-file>
    </welcome-file-list>
</web-app>

 

1.ContextLoaderListener的作用:

ContextLoaderListener監聽器的作用就是啟動Web容器時,自動裝配ApplicationContext的配置信息。

因為它實現了ServletContextListener這個接口,在web.xml配置這個監聽器,啟動容器時,就會默認執行它實現的方法。

在ContextLoaderListener中關聯了ContextLoader這個類,所以整個加載配置過程由ContextLoader來完成。

 

通俗簡單理解:必須要配置這個,用來讀取Spring配置文件

 

2.contextConfigLocation的作用:

配置spring配置文件位置參數,這里我Spring的XML配置文件在src下,直接寫即可

 

3.OpenSessionInViewFilter的作用:

Spring為我們解決Hibernate的Session的關閉與開啟問題。

Hibernate 允許對關聯對象、屬性進行延遲加載,但是必須保證延遲加載的操作限於同一個 Hibernate Session 范圍之內進行。

如果 Service 層返回一個啟用了延遲加載功能的領域對象給 Web 層,當 Web 層訪問到那些需要延遲加載的數據時,由於加載領域對象的 Hibernate Session 已經關閉,這些導致延遲加載數據的訪問異常

比如拋出這種異常:

org.hibernate.LazyInitializationException:(LazyInitializationException.java:42) 
 - failed to lazily initialize a collection of role: cn.easyjava.bean.product.ProductType.childtypes, no session or session was closed
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: cn.easyjava.bean.product.ProductType.childtypes, 
no session or session was closed

 用來把一個Hibernate Session和一次完整的請求過程對應的線程相綁定。目的是為了實現"Open Session in View"的模式。例如: 它允許在事務提交之后延遲加載顯示所需要的對象。

而Spring為我們提供的OpenSessionInViewFilter過濾器為我們很好的解決了這個問題。

OpenSessionInViewFilter的主要功能是用來把一個Hibernate Session和一次完整的請求過程對應的線程相綁定。

目的是為了實現"Open Session in View"的模式。例如: 它允許在事務提交之后延遲加載顯示所需要的對象

OpenSessionInViewFilter 過濾器將 Hibernate Session 綁定到請求線程中,它將自動被 Spring 的事務管理器探測到。

所以 OpenSessionInViewFilter 適用於 Service 層使用HibernateTransactionManager進行事務管理的環境,也可以用於非事務只讀的數據操作中

 

通俗簡單解釋:有時候數據庫查到的結果返回到jsp頁面會拋異常,配置好這個就可以解決。在前端頁面顯示完相應的結果再關閉session

 

其他注意事項:

1.strust2核心過濾器一定要寫在最后邊

2.開頭我這里設置的session有效時間60(單位:分鍾)

 

接下來Spring配置文件:

命名什么都可以,我取名applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns="http://www.springframework.org/schema/beans" xmlns:context="http://www.springframework.org/schema/context"
    xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.2.xsd 
                            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd 
                            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-4.2.xsd 
                            http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.2.xsd ">

    <!-- 讀取db.properties文件 -->
    <context:property-placeholder location="classpath:db.properties" />
    <!-- 配置c3p0連接池 -->
    <bean name="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
        <property name="driverClass" value="${jdbc.driverClass}"></property>
        <property name="user" value="${jdbc.user}"></property>
        <property name="password" value="${jdbc.password}"></property>
    </bean>

    <!-- 核心事務管理器 -->
    <bean name="transactionManager"
        class="org.springframework.orm.hibernate5.HibernateTransactionManager">
        <property name="sessionFactory" ref="sessionFactory"></property>
    </bean>

    <!-- 配置通知 -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="save*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="persist*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="update*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="modify*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="delete*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="remove*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
            <tx:method name="get*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="true" />
            <tx:method name="find*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="true" />
            <tx:method name="*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="false" />
        </tx:attributes>
    </tx:advice>
    <!-- 配置將通知織入目標對象 -->
    <aop:config>
        <aop:pointcut expression="execution(* service.impl.*ServiceImpl.*(..))"
            id="txPc" />
        <aop:advisor advice-ref="txAdvice" pointcut-ref="txPc" />
    </aop:config>

    <!-- 在spring配置中放置hibernate配置信息 -->
    <bean name="sessionFactory"
        class="org.springframework.orm.hibernate5.LocalSessionFactoryBean">
        <!-- 將連接池注入到sessionFactory, hibernate會通過連接池獲得連接 -->
        <property name="dataSource" ref="dataSource"></property>
        <!-- 配置hibernate基本信息 -->
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect
                </prop>

                <!-- 可選配置 -->
                <prop key="hibernate.hbm2ddl.auto">update</prop>
            </props>
        </property>
        <!-- 引入orm元數據,指定orm元數據所在的包路徑,spring會自動讀取包中的所有配置 -->
        <property name="mappingDirectoryLocations" value="classpath:domain"></property>
    </bean>
    <!-- action -->
    <!-- 注意:Action對象作用范圍一定是多例的(prototype).這樣才符合struts2架構 -->
    <bean name="userAction" class="web.action.UserAction" scope="prototype">
        <property name="userService" ref="userService"></property>
    </bean>
    <!-- service -->
    <bean name="userService" class="service.impl.UserServiceImpl">
        <property name="userdao" ref="userDao"></property>
    </bean>
    <!-- dao -->
    <bean name="userDao" class="dao.impl.UserDaoImpl">
        <property name="sessionFactory" ref="sessionFactory"></property>
    </bean>
</beans>

這里解釋下Spring配置文件以及易錯處:

1.數據庫參數我寫在一個db.properties文件種,方便開發中修改

2.接下來配置c3p0連接池,雖然不使用也可以,但是hibernate推薦使用

 

3.transactionManager這個bean,管理事務嵌套,開啟,關閉,資源線程同步,提交,回滾

初步了解, HibernateTransactionManager這個類提供 sessionFactory的管理。

為了實現數據同步,在HibernateTransactionManager內部會進行Hibernate session的open和close,並將打開的Hibernaate sesion關聯到當前的Application session。

在Application中則通過getCurrentSession方式獲取爭取的打開的Hibernate session,  從而解決某些方面的線程安全及同步問題。

 

通俗解釋,這里配置的意義在於:只有配置好這一項,才可以在DAO層使用HibernateTemplate,方便數據庫操作

 

4.配置通知和AOP方面,見我以前文章:

http://www.cnblogs.com/xuyiqing/p/8463598.html

http://www.cnblogs.com/xuyiqing/p/8464465.html

這里大概解釋下:

例如這里:

            <tx:method name="get*" isolation="REPEATABLE_READ"
                propagation="REQUIRED" read-only="true" />

這一句的意思是:所有DAO層方法,在生效前,先通過這里的“通知”,如果是get開頭的方法,只能查詢數據庫,無法修改

這里的兩個參數:isolation="REPEATABLE_READ"事務隔離級別,暫不做解釋,開發中只選擇這一項

propagation="REQUIRED"事務傳播行為,暫不做解釋,開發中只選擇這一項

最后一個屬性,是否只讀:如果是查詢類方法,應該設置為只讀,如果是增刪更新操作,應該設置為false

 

下面:expression="execution(* service.impl.*ServiceImpl.*(..))這里是通配方法,為所有service包的實現類中的所有方法配置切點

后邊是一個增強AOP配置,配好即可,沒什么解釋的

 

5.在spring配置中放置hibernate配置信息:

先注入上面配置好的c3p0連接池,再配置Hibernate

<prop key="hibernate.hbm2ddl.auto">update</prop>的意思是,每一次生成表的時候如果表存在則覆蓋,表不存在的話新建一張表,這種方式是最常用的

<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>使用MySQL方言,我使用MySQL開發,所有使用這個配置

 

<!-- 引入orm元數據,指定orm元數據所在的包路徑,spring會自動讀取包中的所有配置 -->
<property name="mappingDirectoryLocations" value="classpath:domain"></property>

這里的意思:hibernate需要一些元數據配置,我把這些配置寫在了一個domain包中

 

最后的三層結構的bean配置就不多解釋了

 

附:db.properties

jdbc.jdbcUrl=jdbc:mysql:///blog?useUnicode=true&characterEncoding=utf-8&autoReconnect=true
jdbc.driverClass=com.mysql.jdbc.Driver
jdbc.user=root
jdbc.password=123456

url設置的時候需要注意下:這里有個大坑,當時解決了幾天。

后邊加入characterEncoding=utf-8才可以往數據庫中存入中文,否則都是亂碼

后一個參數autoReconnect=true是意思的:自動重連,MySQL如果長時間不操作就會關閉(默認8小時)

 

上邊提到了hibernateORM元數據配置:

這里繼續

我建立了一個實體類:User(get、set方法)

package domain;

public class User {
    private Long u_id;
    private String username;
    private String u_password;
    private String qq;
    private String avatar;
    private Integer article_count;

    private String checkCode;

    
    
    
    public String getCheckCode() {
        return checkCode;
    }

    public void setCheckCode(String checkCode) {
        this.checkCode = checkCode;
    }

    public Long getU_id() {
        return u_id;
    }

    public void setU_id(Long u_id) {
        this.u_id = u_id;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getU_password() {
        return u_password;
    }

    public void setU_password(String u_password) {
        this.u_password = u_password;
    }

    public String getQq() {
        return qq;
    }

    public void setQq(String qq) {
        this.qq = qq;
    }

    public String getAvatar() {
        return avatar;
    }

    public void setAvatar(String avatar) {
        this.avatar = avatar;
    }

    public Integer getArticle_count() {
        return article_count;
    }

    public void setArticle_count(Integer article_count) {
        this.article_count = article_count;
    }
}
View Code

 

對應ORM元數據配置:

User.hbm.xml:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="domain">
    <class name="User" table="blog_user">
        <id name="u_id">
            <generator class="native"></generator>
        </id>
        <property name="username" column="username"></property>
        <property name="u_password" column="u_password"></property>
        <property name="qq" column="qq"></property>
        <property name="avatar" column="avatar"></property>
        <property name="article_count" column="article_count"></property>
    </class>
</hibernate-mapping>

 package是包名:在這個包下找到User類,映射

table是新建的表名字,Id是主鍵生成策略:

具體見我以前文章:http://www.cnblogs.com/xuyiqing/p/8449059.html

實體開發使用native即可

接下來就是配置表的字段名即可

 

復雜的配置,如:一對多,多對多等(外鍵)

 

比如我這里還有一個article文章類:一個用戶可以有多個文章,一個分類下也可以有多個文章:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-mapping PUBLIC 
    "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
    "http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<hibernate-mapping package="domain">
    <class name="Article" table="blog_article">
        <id name="article_id">
            <generator class="native"></generator>
        </id>
        <property name="title"></property>
        <property name="summary"></property>
        <property name="read_count"></property>
        <property name="comment_count"></property>
        <property name="up_count"></property>
        <property name="create_time"></property>
        <property name="article_detail"></property>
        <many-to-one name="article_type" column="article_type_id" class="Articletype"></many-to-one>
        <many-to-one name="author" column="article_author_id" class="User"></many-to-one>
    </class>
</hibernate-mapping>

 

更多的細節見我以前的文章:

http://www.cnblogs.com/xuyiqing/category/1163473.html

 

 

接下來是struts2的配置:

<?xml version="1.0" encoding="UTF-8"?>
  <!DOCTYPE struts PUBLIC
    "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
    "http://struts.apache.org/dtds/struts-2.3.dtd">

<struts>
    <constant name="struts.multipart.maxSize" value="31457280"></constant>
    <constant name="struts.objectFactory" value="spring"></constant>

    <package name="Blog" namespace="/" extends="struts-default">
        <interceptors>
            <interceptor name="privilegeInterceptor" class="web.interceptor.PrivilegeInterceptor"></interceptor>
            <interceptor-stack name="myStack">
                <interceptor-ref name="privilegeInterceptor">
                    <param name="excludeMethods">login,regist,createRandom,execute</param>
                </interceptor-ref>
                <interceptor-ref name="defaultStack"></interceptor-ref>
            </interceptor-stack>
        </interceptors>
        <default-interceptor-ref name="myStack"></default-interceptor-ref>
        <global-results>
            <result name="toHome" type="redirect">/IndexAction_articleTypeList
            </result>
            <result name="toLogin" type="redirect">/login.jsp</result>
        </global-results>

        <!-- 整合:class屬性上填寫spring中action對象的BeanName 完全由spring管理action生命周期,包括Action的創建 -->
        <action name="UserAction_*" class="userAction" method="{1}">
            <exception-mapping result="loginError"
                exception="java.lang.RuntimeException"></exception-mapping>
            <result name="loginError">/login.jsp</result>
            <result name="regist">/register.jsp</result>
        </action>
        <action name="CheckImgAction" class="web.action.CheckImgAction">
            <result name="success" type="stream">
                <param name="contentType">image/jpeg</param>
                <param name="inputName">inputStream</param>
            </result>
        </action>
        <action name="UploadAction_*" class="uploadAction" method="{1}">
            <result name="success">/index.jsp</result>
               <interceptor-ref name="defaultStack">
                   <param name="fileUpload.allowedExtensions">jepg,jpg,gif,png</param>
               </interceptor-ref> 
            <result name="input">/index.jsp</result>
        </action>
    </package>

    <package name="article" extends="json-default">
        <action name="OthersAction_*" class="othersAction" method="{1}">
            <result name="good" type="json"></result>
            <result name="reply" type="json"></result>
            <result name="delete" type="json"></result>
        </action>
    </package>
</struts>
    

 

開頭配置了兩個常量:上次文件最大字節、交給Spring管理Action

后邊的:

1.這里我設置了一個攔截器:暫且稱它為登錄狀態檢驗器:如果不登陸,無法訪問我的BBS,下邊我寫出攔截器代碼,這里先看配置

要把自定義攔截器放在默認攔截器前面

                    <param name="excludeMethods">login,regist,createRandom,execute</param>

這里的意思是不攔截登錄、注冊、隨機生成驗證碼方法

2.接下來配置的是全局結果集

3.后邊的Action配置就不必詳細說了:注意這一句

            <exception-mapping result="loginError"
                exception="java.lang.RuntimeException"></exception-mapping>

這里我為什么要配置一個異常呢?非常有用

用於登錄錯誤回顯,Exception里面可以放message,即錯誤信息,比如用戶名不存在等等

如果有異常,這里就返回當前頁面,帶着錯誤信息,前端頁面用EL或struts2標簽顯示即可:例如

<s:property value="exception.message" />

 

4.后邊的Action參數配置還可以是一個流對象:用於做驗證碼圖片臨時保存

5.后邊我留下了一個文件上傳Action的配置:

<interceptor-ref name="defaultStack">
<param name="fileUpload.allowedExtensions">jepg,jpg,gif,png</param>
</interceptor-ref>

這里的意思是:我做了一個過濾器,只可以上傳圖片文件

6.<package name="article" extends="json-default">

這個包的作用很重要:是專門處理AJAX請求的,struts-default無法處理AJAX請求

 

這里是登錄攔截器的代碼:

package web.interceptor;

import java.util.Map;

import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.interceptor.MethodFilterInterceptor;

import domain.User;

//驗證登錄狀態攔截器
@SuppressWarnings("all")
public class PrivilegeInterceptor extends MethodFilterInterceptor {

    @Override
    // 不攔截登陸和注冊方法
    protected String doIntercept(ActionInvocation invocation) throws Exception {
        // 1 獲得Session
        Map<String, Object> session = ActionContext.getContext().getSession();
        // 2 獲得登陸標識
        User user = (User) session.get("user");
        // 3 判斷標識是否存在
        
        if (user != null) {
            // 存在=> 放行
            return invocation.invoke();
        } else {
            // 不存在=> 重定向到登陸頁面
            return "toLogin";
        }    

    }

}

 

寫的很詳細,不必多說,都能看懂吧

 

配置文件部分內容就完了,后邊陸續更新代碼方面的坑和注意事項

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM