学习SSH,SSM框架有感

JSP/Servlet基础

许多同学学习框架的时候,往往会忽视这部分内容,有的甚至没学JSP直接学习了框架,导致基础知识不扎实,只是会用框架,最后连JSP一共有几个内置对象都不知道。自己当时学的时候也没怎么学这部分知识,就开始学Struts了,也是最近几天又认真地把这部分知识看了一遍。
JSP和Servlet,区别上一个是Servlet没有内置对象,还有就是JSP可以在HTML中嵌入Java代码。而实际上,我们都知道,JSP最后也会被翻译为Servlet.所以广义的Servlet也可以指JSP+Servlet. 而JSP中的7个动作指令,9种内置对象,还有EL表达式,以及EL表达式的11个内置对象。都是我们必须要掌握的。

1.关于Cookie

Cookie用于网站记录用户的某些信息,它和session的区别在于:session是存放在服务器中,超出服务器设置的失效时间就会失效,而Cookie会一直存放在客户端机器上,除非超出Cookie的生命期限。增加Cookie也是使用response内置对象完成的。事实上,用来实现session的sessionId也是存放在Cookie上,如果浏览器禁用了Cookie,同样session也会失效。

(1)jQuery中操作Cookie

jQuery中操作Cookie非常简单,要先下载jQuery的Cookie插件jquery.cookie.js.
增加Cookie:$.cookie("c1","zhazha"); 将值"zhzha"写入cookie名为c1的cookie中。
$.cookie后面还有其他参数,用来设置有效期,路径等。
例如:$.cookie("c1","zhazha",{path: "/", expiress: 7}); 创建一个有效期为7天,路径与创建页路径一致的cookie.
删除Cookie:$.cookie("c1",NULL); 销毁名称为c1的cookie.
得到Cookie:$.cookie("c1"); 得到cookie名为c1的cookie的value值。

  • 注:对于Chrome,使用本地测试时,cookie是基于域名来储存的。要放到测试服务器上或者本地localhost服务器上才会生效。cookie具有不同域名下储存不可共享的特性。单纯的本地一个html页面打开获取cookie是无效的。可以使用IE浏览器或者火狐浏览器。

(2)利用response增加,删除Cookie

Cookie c=new Cookie("username","name");
c.setMaxAge(24 * 3600);//设置生存期为24小时
response.addCookie(c);//增加Cookie
c.setMaxAge(0);//删除Cookie


Struts

Struts是Web框架,学习Struts首先要理解MVC思想,即Model(模型)、View(视图)和Controller(控制器)。学习Struts时间最长,也是在spring mvc之前用得最多的。Struts 2框架本身大致可以分为3个部分:核心控制器FilterDispatcher、业务控制器Action和用户实现的企业业务逻辑组件。三个组件相互配合,完成各个逻辑功能。

1.Struts的MVC结构

刚想学习Struts的时候,常常有人要把之前用的三层结构和MVC对应起来。其实没有必要,之前的三层架构和MVC结构还是有区别的。那么Struts的MVC分别指什么呢。
Model模型:这里的Model不只指我们项目Model层里的POJO对象。还包括Action对象,以及action调用的业务逻辑组件。
View视图:显然,视图就是指我们的JSP页面,以及Struts提供的标签库能够与我们的ActionFrom(包括request,session)进行交互。
Controller控制器:控制其主要就是指Struts的ActionServlet和ActionMapping,也就是我们配置的struts.xml.

2.ValueStack和StackContext

学习OGNL的时候,对这两个概念经常搞不清,总容易混淆。
ValueStack 称之为值栈,可以理解为StackContext的根对象,在Action里,要有set和get方法。在使用OGNL时,若访问的属性属于根对象,可以直接访问该属性,不用加'#'.
StackContext 叫做Stack的上下文,是整个OGNL的计算和求值的Context. 除了包含ValueStack这个根对象外,还包含一系列对象,包括request/session/application等。而访问这些命名对象时,要加'#'.

3.jstl和struts标签库

一直纠结这两个到底哪个好用,当然struts标签库只用应用与struts项目中。先学习过jstl,后来做项目的时候又一直用的struts的标签库,用起来感觉没什么差别,后来结合富文本编辑器从后台取数据的时候才发现,二者的取值方式不同。jstl取值的时候会转换里面的html标签,而struts标签取值的时候只是把值当做字符串输出到页面。(后来发现struts标签也可以通过设置escape='1',对html字符经行转义,当然jstl也可设置对html不进行转义,只是二者默认方式不同。但是Struts标签的访问速度要比jstl慢,效率低)

4.Struts的Ajax支持

自己做项目的时候最多用的就是jQuery中的Ajax的post函数,来实现异步请求。
用法:$.post("",{},function(data,textStatus){})
第一个引号里面传一个URL,一般就是一个action路径,第二个{}中传一个JSON对象,如{a:1,b:2}这样,把这个对象传给后台Action,后台通过set和get方法得到a和b的值,第三个function是方法实现体。用来处理后台传送回的数据和前台经行交互。
后台通过response.getWriter.print("")来向前台传递数据,存放到function中的data里。这样就实现了简单的Ajax的异步请求。

Hibernate

Hibernate是位于持久层的ORM框架,ORM即对象/关系数据库映射。Hibernate工作就是将数据库中的关系型数据映射成Java对象。
由于Hibernate过于厚重,学习成本高,导致其逐渐被Mybatis取代。
当然Hibernate也有优点,比如一套hql可以适配所有数据库,还有其缓存机制等都是优点。
由于Hibernate门槛较高,还是建议直接学习Mybatis.

Spring

作为应用范围最广的容器框架,无论走到哪基本都会用到Spring. Spring可以说是企业应用开发的不二之选,Spring贯穿表现层,业务层,持久层。并且能以高度的开放性与他们无缝整合。
Spring的核心是IOC(控制反转),DI(依赖注入)和AOP(面向切面的编程)。实际上,IOC和DI是同一个概念,只是spring设计者认为DI能更准确表示Spring核心技术。

1.IOC/DI

IOC的思想最核心的地方在于,资源不由使用资源的双方管理,而由不使用资源的第三方管理,这可以带来很多好处。第一,资源集中管理,实现资源的可配置和易管理。第二,降低了使用资源双方的依赖程度,也就是耦合度。

2.AOP

AOP思想能从程序运行角度考虑程序的流程,提取业务处理过程的切面。其有着各个步骤之间有良好的隔离性,源代码无关性的特性。可以这样理解,可以把代码看成一坨面,同样用刀来切面,切完了,想在前面、中间、后面或者四周放些“香料”什么的随便你。你只要把这把刀实现好就行了。AOP面向的是程序运行中各个步骤,希望以更好的方式来组合业务处理的各个步骤。

SSH 整合开发

1.Spring整合Struts2

Spring整合Struts2有两种策略,一种是让Spring管理控制器,一种是使用Spring的自动装备。两种方式各有优缺点。本人应用的是第二种,即使用自动装备策略。

(1)首先载入配置文件

在web.xml中加入以下信息。当然如果你使用IDEA开发时,这些都是已经自动生成好的。

<context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/applicationContext.xml</param-value>
</context-param>
<listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>

(2)配置struts.xml

由于使用自动装配策略,所以struts.xml的配置方式和原来一模一样。即和在没有Spring整合之前的配置方式一样。

(3)注入Action中的Service组件。

在applicationContext.xml中配置Service的Bean,其中Bean的id与Action定义的Service中其getService中的名字一样。例如:

 public UserService us;
    public UserService getUs() {
        return us;
    }
    public void setUs(UserService us) {
        this.us = us;
    }

如果定义了这样一个Service,那么bean id也应该设置为us,即:

<bean id="us" class="com.service.serviceimpl.UserServiceImpl"/>

2.Spring整合Hibernate

(1)配置管理SessionFactory.

在applicationContext.xml中配置以下信息,例子中包含两个POJO,即User和Friend.

<!--使用c3p0配置数据源-->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
          destroy-method="close"
          p:driverClass="com.mysql.jdbc.Driver"
          p:jdbcUrl="jdbc:mysql://localhost/friends"
          p:user="root"
          p:password="123"
          p:maxPoolSize="40"
          p:minPoolSize="2"
          p:initialPoolSize="2"
          p:maxIdleTime="30"/>
    <!--配置SeesionFactory,注入数据源(dataSource)-->
    <bean id="sessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean"
    p:dataSource-ref="dataSource">
        <property name="annotatedClasses">
            <list>
                <value>com.models.User</value>
                <value>com.models.Friend</value>
            </list>
        </property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">
                    org.hibernate.dialect.MySQL5InnoDBDialect
                </prop>
                <prop key="hibernate.hbm2ddl.auto">
                    update
                </prop>
            </props>
        </property>
    </bean>

(2)定义DAO组件,并将SessionFactory注入DAO组件。

以下代码含有两个DAO组件,即UserDao和FriendDao.

    <bean id="ud" class="com.dao.impl.UserDaoImpl" p:sessionFactory-ref="sessionFactory"/>
    <bean id="fd" class="com.dao.impl.FriendDaoImpl" p:sessionFactory-ref="sessionFactory"/>

(3)定义Service组件,并将DAO组件注入Service组件。

同理,以下代码含有两个Service组件,UserService和FriendService

    <bean id="us" class="com.service.impl.UserServiceImpl" p:ud-ref="ud" p:fd-ref="fd"/>
    <bean id="fs" class="com.service.impl.FriendServiceImpl" p:fd-ref="fd"/>

(4)配置事务管理器

注:若写入tx报错,说明你没有导入相应的xmlns.

<bean id="transactionManager" class="org.springframework.orm.hibernate4.HibernateTransactionManager"
    p:sessionFactory-ref="sessionFactory"/>
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="get*" read-only="true"/>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice>
    <aop:config>
        <aop:pointcut id="myPointcut" expression="bean(us)||bean(fs)"/>
        <aop:advisor advice-ref="txAdvice" pointcut-ref="myPointcut"/>
    </aop:config>

这样只要在Action中定义Service组件,在Service中定义DAO组件,生成set和get方法,这样,一个SSH项目就搭建好了。

Mybatis

相对于厚重的Hibernate, Mybatis就比较好掌握了。就算没看过Hibernate的新手,看看文档,读读Demo,两天基本就可以掌握上手了。Mybatis实际上是一个不完全的ORM框架,他主要完成的可以说是输入和输出映射。需要手动写sql语句,就简单的增删改查来说,个人更喜欢Hibernate. 觉得比Mybatis操作简单,能让让人更加专注于写业务逻辑。而学习了Mybatis的高级映射和缓存机制后,确实感觉很简单,比Hibernate容易理解。尤其是学完逆向工程之后,发现Mybatis也可以如此轻松。

1.Mybatis与Hibernate的比较

Mybatis优势
(1).MyBatis可以进行更为细致的SQL优化,可以减少查询字段。
(2).MyBatis容易掌握,而Hibernate门槛较高。
Hibernate优势
(1).Hibernate的DAO层开发比MyBatis简单,Mybatis需要维护SQL和结果映射。
(2).Hibernate对对象的维护和缓存要比MyBatis好,对增删改查的对象的维护要方便。
(3).Hibernate数据库移植性很好,MyBatis的数据库移植性不好,不同的数据库需要写不同SQL。
(4).Hibernate有更好的二级缓存机制,可以使用第三方缓存。MyBatis本身提供的缓存机制不佳。
他人总结
(1).Hibernate功能强大,数据库无关性好,O/R映射能力强,如果你对Hibernate相当精通,而且对Hibernate进行了适当的封装,那么你的项目整个持久层代码会相当简单,需要写的代码很少,开发速度很快,非常爽。
(2).Hibernate的缺点就是学习门槛不低,要精通门槛更高,而且怎么设计O/R映射,在性能和对象模型之间如何权衡取得平衡,以及怎样用好Hibernate方面需要你的经验和能力都很强才行。
(3).Mybatis入门简单,即学即用,提供了数据库查询的自动对象绑定功能,而且延续了很好的SQL使用经验,对于没有那么高的对象模型要求的项目来说,相当完美。
(4).Mybatis的缺点就是框架还是比较简陋,功能尚有缺失,虽然简化了数据绑定代码,但是整个底层数据库查询实际还是要自己写的,工作量也比较大,而且不太容易适应快速数据库修改。

  • 注:以上内容摘自知乎

2.Mybatis的分页

分页是Mybatis的一个弱项,不能像Hibernate那样直接有分页的方法供你调用。而自己通过写sql语句来写分页也是个苦力活,很麻烦,所以我强烈向大家推荐一款插件:PageHelper. 来自OSChina社区的某位自己写的插件,无论是配置还是操作都超级简单,爱不释手。

3.Mybatis的逆向工程

Mybatis需要我们自己写sql语句,每张表的增删改查无疑给我们带来了巨大的工作量,所以项目中一般使用Mybatis的逆向工程工具,自动生成Java代码,为我们提供丰富的操作数据库的功能。

(1)使用之前需要先下载用于支持逆向工程的jar包

点击下载,网上的方法都是基于Maven或者用的是Eclipse的插件。自己使用IDEA开发的,所以用的是第三种方法,通过配置xml文件,通过运行它提供主程序生成Java代码。当然,用第一种方法也行。

(2)配置generatorConfig.xml

配置要求以下代码中注释写的很清楚

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE generatorConfiguration
  PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN"
  "http://mybatis.org/dtd/mybatis-generator-config_1_0.dtd">

<generatorConfiguration>
    <context id="testTables" targetRuntime="MyBatis3">
        <commentGenerator>
            <!-- 是否去除自动生成的注释 true:是 : false:否 -->
            <property name="suppressAllComments" value="true" />
        </commentGenerator>
        <!--数据库连接的信息:驱动类、连接地址、用户名、密码 -->
        <jdbcConnection driverClass="com.mysql.jdbc.Driver"
            connectionURL="jdbc:mysql://localhost:3306/friends" userId="root"
            password="123">
        </jdbcConnection>

        <javaTypeResolver>
            <property name="forceBigDecimals" value="false" />
        </javaTypeResolver>

        <!-- targetProject:生成PO类的位置 -->
        <javaModelGenerator targetPackage="com.models"
            targetProject=".\src">
            <!-- enableSubPackages:是否让schema作为包的后缀 -->
            <property name="enableSubPackages" value="false" />
            <!-- 从数据库返回的值被清理前后的空格 -->
            <property name="trimStrings" value="true" />
        </javaModelGenerator>
        <!-- targetProject:mapper映射文件生成的位置 -->
        <sqlMapGenerator targetPackage="com.mapper"
            targetProject=".\src">
            <!-- enableSubPackages:是否让schema作为包的后缀 -->
            <property name="enableSubPackages" value="false" />
        </sqlMapGenerator>
        <!-- targetPackage:mapper接口生成的位置 -->
        <javaClientGenerator type="XMLMAPPER"
            targetPackage="com.mapper"
            targetProject=".\src">
            <!-- enableSubPackages:是否让schema作为包的后缀 -->
            <property name="enableSubPackages" value="false" />
        </javaClientGenerator>
        <!-- 指定数据库表 -->
        <table tableName="user"></table>
        <table tableName="friend"></table>
    </context>
</generatorConfiguration>

(3)运行GeneratorSqlmap

以下代码只需要修改generatorConfig.xml的位置

package com.service;

import java.io.File;
import java.util.ArrayList;
import java.util.List;

import org.mybatis.generator.api.MyBatisGenerator;
import org.mybatis.generator.config.Configuration;
import org.mybatis.generator.config.xml.ConfigurationParser;
import org.mybatis.generator.internal.DefaultShellCallback;

public class GeneratorSqlmap {

    public void generator() throws Exception{

        List<String> warnings = new ArrayList<String>();
        boolean overwrite = true;
        //指定 逆向工程配置文件
        File configFile = new File("generatorConfig.xml");
        ConfigurationParser cp = new ConfigurationParser(warnings);
        Configuration config = cp.parseConfiguration(configFile);
        DefaultShellCallback callback = new DefaultShellCallback(overwrite);
        MyBatisGenerator myBatisGenerator = new MyBatisGenerator(config,
                callback, warnings);
        myBatisGenerator.generate(null);

    } 
    public static void main(String[] args) throws Exception {
        try {
            GeneratorSqlmap generatorSqlmap = new GeneratorSqlmap();
            generatorSqlmap.generator();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

这样所有的model和mapping就自动生成好了,具体的用法看代码,有的看方法名就能直接看出他的用法,其中的selectByExample类似于Hibernate的Criteria, 也就是按条件查询。用法也很简单,就不演示了。

Spring mvc

作为替代Struts2的另一个web框架,Spring mvc当然有它的优势。刚接触Spring mvc时,为了理解Spring mvc的结构,要接触许多名词,往往搞得我们头疼。包括前端控制器,处理器映射器,处理器适配器,以及视图解析器。所以我们首先要了解Spring mvc的工作流程。

1.Spring mvc的工作流程

(1)浏览器发起请求到前端控制器(DispatcherServlet)
(2)前端控制器请求HandlerMapping查找 Handler
(3)处理器映射器HandlerMapping向前端控制器返回Handler
(4)处理器适配器执行Handler并返回ModelAndView
(5)视图解析器向前端控制器返回View
(6)前端控制器进行视图渲染

2.关于@RequestParam

真正做项目时,@RequestParam是必须配置的,即使你传入的参数名和你的形参名一样也配置,因为在Eclipse的Debug模式下编译时,参数名都会保留在class文件中,spring由此可以反射绑定。而自己使用的IDEA,无论怎么调试怎样都得不到前面传过来的值,调了很长时间,只有加上@RequestParam(value = "username") String username这样的同名注解才行,即使文档里说可以省略。

3.Spring mvc的优势

(1)学习完Spring mvc之后可以发现,Spring mvc是基于于方法开发的,所有需要的参数都通过Controller里面的方法形参进行传递,而Struts是基于于类开发的。这种基于方法的Controller开发更类似于Service开发,逻辑上更符合我们的编程逻辑。
(2)由于Struts的标签库,导致Struts的效率要低于Spring mvc.
(3)Struts有安全漏洞,Spring mvc没有发现安全隐患。

SSM 整合开发

项目结构:src下一共四个包,com.models, com.mapper, com.service, com.controller. 其中com.models包含两个POJO对象User和Friend. com.models和com.mapper均由mybatis逆向工程自动生成。

1.配置web.xml

这个很简单,如果使用IDEA开发,这些都是自动生成好的。例子中除了Mapper的xml一共包含三个xml,即WEB-INF下的applicationContext.xml和dispatcher-servlet.xml. 以及src下的sqlMapConfig.xml.

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
         version="3.1">
    <context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>/WEB-INF/applicationContext.xml</param-value>
    </context-param>
    <listener>
        <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
    </listener>
    <servlet>
        <servlet-name>dispatcher</servlet-name>
        <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
        <load-on-startup>1</load-on-startup>
    </servlet>
    <servlet-mapping>
        <servlet-name>dispatcher</servlet-name>
        <url-pattern>*.form</url-pattern>
    </servlet-mapping>
    <servlet-mapping>
        <servlet-name>dispatcher</servlet-name>
        <url-pattern>*.action</url-pattern>
    </servlet-mapping>
</web-app>

2.配置sqlMapConfig.xml

这里我只给model配置了一个别名,可以根据需要自行配制,例如可以配置二级缓存。

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration>
    <!--配置别名-->
    <typeAliases>
        <package name="com.models"/>
    </typeAliases>
</configuration>

3.配置applicationContext.xml

这里的配置方式和整合SSH的配置方式类似

<!--使用c3p0配置数据源-->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"
          destroy-method="close"
          p:driverClass="com.mysql.jdbc.Driver"
          p:jdbcUrl="jdbc:mysql://localhost/friends"
          p:user="root"
          p:password="123"
          p:maxPoolSize="40"
          p:minPoolSize="2"
          p:initialPoolSize="2"
          p:maxIdleTime="30"/>

    <!--配置sqlSessionFactory-->
    <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="configLocation" value="classpath:sqlMapConfig.xml"/>
    </bean>

    <bean id="userMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
        <property name="mapperInterface" value="com.mapper.UserMapper"/>
        <property name="sqlSessionFactory" ref="sqlSessionFactory"/>
    </bean>
    <bean id="friendMapper" class="org.mybatis.spring.mapper.MapperFactoryBean">
        <property name="mapperInterface" value="com.mapper.FriendMapper"/>
        <property name="sqlSessionFactory" ref="sqlSessionFactory"/>
    </bean>
    <bean id="us" class="com.service.impl.UserServiceImpl" p:userMapper-ref="userMapper"/>
    <bean id="fs" class="com.service.impl.FriendServiceImpl" p:friendMapper-ref="friendMapper"/>
    <!--配置事务管理器-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="save*" propagation="REQUIRED"/>
            <tx:method name="delete*" propagation="REQUIRED"/>
            <tx:method name="insert*" propagation="REQUIRED"/>
            <tx:method name="update*" propagation="REQUIRED"/>
            <tx:method name="find*" propagation="SUPPORTS" read-only="true"/>
            <tx:method name="get*" propagation="SUPPORTS" read-only="true"/>
            <tx:method name="select*" propagation="SUPPORTS" read-only="true"/>
        </tx:attributes>
    </tx:advice>
    <aop:config>
        <aop:pointcut id="myPointcut" expression="bean(us)||bean(fs)"/>
        <aop:advisor advice-ref="txAdvice" pointcut-ref="myPointcut"/>
    </aop:config>

3.配置dispatcher-servlet.xml

    <!--配置Handler-->
    <bean class="com.controller.UserController" p:us-ref="us"/>
    <bean class="com.controller.FriendController" p:fs-ref="fs"/>
    <!--配置使用注解开发方式-->
    <mvc:annotation-driven></mvc:annotation-driven>
    <!--配置视图解析器-->
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"/>

这样配置好之后,同样的,只要在Service层添加Mapper对象,在Controller层添加Service对象,这样一个SSM项目就搭建好了。

总结:

利用这个假期把Spring,Mybatis,Spring mvc过了一遍,有些感悟与体会。希望上面的内容能对看的人有所帮助,还有上面关于概念的理解与对比也是经常被问,经常让人困惑的,愿学有所成。

Comments
Write a Comment