Spring MVC快速上手教程

Spring Framework可以被使用在很多场合之中,考虑到目前大多数Java EE的项目是B/S结构的,所以这里的快速上手教程会以Spring MVC为切入点,用最简单的代码一步一步来实现一个图书列表的页面。

在正式动手之前需要做一些准备工作,先安装并设置好JDK 1.5和Tomcat 5,关于数据库及其访问方式可以根据个人习惯进行选择,教程中使用MySQL数据库和Hibernate(映射由Hibernate Annotation实现)。请将实际使用到的jar文件复制到WEB-INF/lib目录中,整个项目的结构见图1,教程中用到的jar文件见图2。

结构及用到的Jar文件

项目中的Bean定义分散在多个XML文件中,每完成一部分代码就给出相应的配置,最后再进行整合和部署。配置中使用default-autowire="byName"实现了Bean的自动织入,节省了很多个工作量,只需注意Bean及属性的命名即可。

Step 1.Business Objects & DAO

教程中的例子涉及到两个实体对象,代表文章的Article类和代表作者的Author类,分别对应了数据库中的article表和author表,一篇文章有一个作者,而一个作者可以有多篇文章。类的代码如下(省略getter和setter):

代码:Article.java

None.gif package demo.model;
None.gif
None.gif import javax.persistence. * ;
None.gif
None.gif@Entity
ExpandedBlockStart.gifContractedBlock.gif public class Article ... {
InBlock.gif @Id
InBlock.gif @GeneratedValue
InBlock.gifprivate Long id;
InBlock.gif
InBlock.gifprivate String title;
InBlock.gif
InBlock.gif @ManyToOne
InBlock.gifprivate Author author;
InBlock.gif
ExpandedBlockEnd.gif}

代码:Author.java

None.gif package demo.model;
None.gif
None.gif import java.util.List;
None.gif import javax.persistence. * ;
None.gif
None.gif@Entity
ExpandedBlockStart.gifContractedBlock.gif public class Author ... {
InBlock.gif @Id
InBlock.gif @GeneratedValue
InBlock.gifprivate Long id;
InBlock.gif
InBlock.gifprivate String name;
InBlock.gif
InBlock.gif @OneToMany
InBlock.gifprivate List<Article> articles;
InBlock.gif
ExpandedBlockEnd.gif}

None.gif

在MySQL中创建数据表的SQL语句如下,数据请自行添加(如果使用Hibernate,表可以根据映射自动生成,具体做法请参考Hibernate文档):

代码:数据库创建SQL

None.gif CREATE DATABASE `articles` DEFAULT CHARACTER SET utf8 COLLATE utf8_general_ci;
None.gif USE articles;
None.gif
None.gif CREATE TABLE `article` (
None.gif `id` bigint ( 20 ) NOT NULL auto_increment,
None.gif `title` varchar ( 100 ) NOT NULL default '' ,
None.gif `author_id` bigint ( 20 ) NOT NULL default ' 0 ' ,
None.gif PRIMARY KEY (`id`)
None.gif) ENGINE = MyISAM DEFAULT CHARSET = utf8;
None.gif
None.gif CREATE TABLE `author` (
None.gif `id` bigint ( 20 ) NOT NULL auto_increment,
None.gif `name` varchar ( 100 ) NOT NULL default '' ,
None.gif PRIMARY KEY (`id`)
None.gif) ENGINE = MyISAM DEFAULT CHARSET = utf8;
None.gif

考虑到可能会有多种DAO的实现,所以在DAO层先定义一个IArticleDao接口,随后可以自由选择具体的实现方式,此处结合Spring的HibernateDaoSupport使用Hibernate来进行实现:

代码:IArticleDao.java

None.gif package demo.dao;
None.gif
None.gif import java.util.List;
None.gif
None.gif import demo.model.Article;
None.gif
ExpandedBlockStart.gifContractedBlock.gif public interface IArticleDao ... {
InBlock.gifpublic List<Article> loadAllArticles();
ExpandedBlockEnd.gif}

None.gif

代码:ArticleDao.java

None.gif package demo.dao;
None.gif
None.gif import java.util.List;
None.gif import org.springframework.orm.hibernate3.support.HibernateDaoSupport;
None.gif import demo.model.Article;
None.gif
ExpandedBlockStart.gifContractedBlock.gif public class ArticleDao extends HibernateDaoSupport implements IArticleDao ... {
InBlock.gif
InBlock.gif @SuppressWarnings("unchecked")
ExpandedSubBlockStart.gifContractedSubBlock.gifpublic List<Article> loadAllArticles() ...{
InBlock.gifreturn (List<Article>)getHibernateTemplate().loadAll(Article.class);
ExpandedSubBlockEnd.gif }

InBlock.gif
ExpandedBlockEnd.gif}

None.gif

接下来对Hibernate进行相应的配置,如果使用了JDO或者iBatis,请参考Spring文档。applicationContext-dao.xml内容如下:

代码:applicationContext-dao.xml

None.gif xml version="1.0" encoding="UTF-8" ?>
None.gif < beans
None.gif xmlns ="http://www.springframework.org/schema/beans"
None.gif xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance"
None.gif xsi:schemaLocation ="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd"
None.gif default-autowire ="byName" >
None.gif <!-- DAO配置于此 --&gt
None.gif < bean id ="articleDao" class ="demo.dao.ArticleDao" />
None.gif
None.gif <!-- 数据源 --&gt
None.gif <!-- JNDI数据源 --&gt
None.gif <!--
None.gif
None.gif
None.gif
None.gif --&gt
None.gif
None.gif <!-- JDBC数据源 --&gt
None.gif < bean id ="dataSource" class ="org.apache.commons.dbcp.BasicDataSource" destroy-method ="close" >
None.gif < property name ="driverClassName" value ="${datasource.jdbc.driverClassName}" />
None.gif < property name ="url" value ="${datasource.jdbc.url}" />
None.gif < property name ="username" value ="${datasource.jdbc.username}" />
None.gif < property name ="password" value ="${datasource.jdbc.password}" />
None.gif bean >
None.gif
None.gif <!-- 使用Annotation映射的sessionFactory --&gt
None.gif < bean id ="sessionFactory" class ="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean" >
None.gif < property name ="dataSource" ref ="dataSource" />
None.gif < property name ="hibernateProperties" >
None.gif < props >
None.gif < prop key ="hibernate.dialect" > ${hibernate.dialect} prop >
None.gif < prop key ="hibernate.show_sql" > ${hibernate.show_sql} prop >
None.gif < prop key ="hibernate.cache.use_query_cache" > ${hibernate.cache.use_query_cache} prop >
None.gif < prop key ="hibernate.cache.provider_class" > ${hibernate.cache.provider_class} prop >
None.gif props >
None.gif property >
None.gif < property name ="annotatedClasses" >
None.gif < list >
None.gif < value > demo.model.Article value >
None.gif < value > demo.model.Author value >
None.gif list >
None.gif property >
None.gif bean >
None.gif
None.gif <!-- 事务管理器,此处为Hibernate的事务管理器 --&gt
None.gif < bean id ="transactionManager" class ="org.springframework.orm.hibernate3.HibernateTransactionManager" />
None.gif beans >
None.gif

此处如果使用JNDI提供数据源,请根据注释进行调整。Spring的事务管理需要声明事务管理器,由于Hibernate、JDO、JDBC的事务管理器都不一样,所以将其与其他事务的配置分开存放。此外,配置中的一些参数使用了占位符(形如${}),这些内容将在Step 4中进行加载。

Step 2.Service

Service层只是调用DAO中的方法为控制器提供图书列表,Service最好能先给出接口,随后进行实现,但此处的功能比较简单,就直接进行实现了:

代码:ArticleService.java

None.gif package demo.service;
None.gif
None.gif import java.util.List;
None.gif import demo.dao.IArticleDao;
None.gif import demo.model.Article;
None.gif
ExpandedBlockStart.gifContractedBlock.gif public class ArticleService ... {
InBlock.gifprivate IArticleDao articleDao;
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gifpublic List<Article> loadAllArticles() ...{
InBlock.gifreturn articleDao.loadAllArticles();
ExpandedSubBlockEnd.gif }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gifpublic void setArticleDao(IArticleDao articleDao) ...{
InBlock.gifthis.articleDao = articleDao;
ExpandedSubBlockEnd.gif }

ExpandedBlockEnd.gif}

None.gif

Spring通过setArticleDao方法为ArticleService注入DAO,也可以选择通过构造方法注入,2.5中还能用@Autowired进行注入。

applicationContext-services.xml内容如下:

代码:applicationContext-services.xml

None.gif xml version="1.0" encoding="UTF-8" ?>
None.gif < beans
None.gif xmlns ="http://www.springframework.org/schema/beans"
None.gif xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance"
None.gif xsi:schemaLocation ="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd"
None.gif default-autowire ="byName" >
None.gif <!-- Service配置于此 --&gt
None.gif < bean id ="articleService" class ="demo.service.ArticleService" />
None.gif beans >
None.gif

Step 3.Controller & View

Spring MVC提供了多种实现控制器的方式,此处直接实现Controller接口,开发一个单一动作的简单控制器,从Service中取得图书列表,提供给视图进行呈现,ListArticleController内容如下:

代码:ListArticleController.java

None.gif package demo.controller;
None.gif
None.gif import java.util.List;
None.gif
None.gif import javax.servlet.http.HttpServletRequest;
None.gif import javax.servlet.http.HttpServletResponse;
None.gif
None.gif import org.springframework.web.servlet.ModelAndView;
None.gif import org.springframework.web.servlet.mvc.Controller;
None.gif
None.gif import demo.model.Article;
None.gif import demo.service.ArticleService;
None.gif
ExpandedBlockStart.gifContractedBlock.gif public class ListArticleController implements Controller ... {
InBlock.gifprivate ArticleService articleService;
InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gifpublic ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception ...{
InBlock.gif List<Article> articles = articleService.loadAllArticles();
InBlock.gif ModelAndView mav = new ModelAndView();
InBlock.gif mav.addObject(articles);
InBlock.gifreturn mav;
ExpandedSubBlockEnd.gif }

InBlock.gif
ExpandedSubBlockStart.gifContractedSubBlock.gifpublic void setArticleService(ArticleService articleService) ...{
InBlock.gifthis.articleService = articleService;
ExpandedSubBlockEnd.gif }

ExpandedBlockEnd.gif}

None.gif

ModelAndView中保存了要传递给视图的对象和具体要使用的视图文件,自2.0起, Spring MVC提供了Convention over Configuration的机制,大大简化了代码与配置。简单地说,名字以Controller结尾的控制器类都会被映射为相应的地址,ListArticleController对应/listarticle*,如果是MultiActionController则会被映射为一个目录;向ModelAndView添加对象时可以不用指定键(key),单一对象的键取决于类名,比如x.y.User的键是user,而某一类对象的Set、List或数组则稍有些复杂,取第一个对象的类名加上“List”作为它的键,比如这里的articles是一个存放Article对象的List,它的键就是articleList;具体的视图会根据请求自动在指定目录中寻找对应的视图文件,本例中就会寻找listarticle(后缀由配置文件决定)。关于Convention over Configuration还有些别的细节,请参考Spring文档的相关章节。

此处的视图比较简陋,只是一张表格,显示了图书的编号、书名和作者,使用JSTL的标签来遍历列表,具体代码如下:

代码:listarticle.jsp

ExpandedBlockStart.gif ContractedBlock.gif ...@ page pageEncoding="UTF-8"%>
ExpandedBlockStart.gifContractedBlock.gif...@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>
None.gif
None.gif<html>
None.gif<head>
None.gif<title>Article Listtitle>
None.gifhead>
None.gif<body>
None.gif<table width="80%" cellspacing="0" cellpadding="0" border="1">
None.gif<thead>
None.gif<tr align="center">
None.gif<td width="20%">编号td><td width="50%">书名td><td width="30%">作者td>
None.giftr>
None.gifthead>
None.gif<tbody>
None.gif<c:forEach items="${articleList}" var=&q

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/7558084/viewspace-1002581/,如需转载,请注明出处,否则将追究法律责任。

上一篇: JSP与Servlet
user_pic_default.png
请登录后发表评论 登录
全部评论
<%=items[i].createtime%>

<%=items[i].content%>

<%if(items[i].items.items.length) { %>
<%for(var j=0;j
<%=items[i].items.items[j].createtime%> 回复

<%=items[i].items.items[j].username%>   回复   <%=items[i].items.items[j].tousername%><%=items[i].items.items[j].content%>

<%}%> <%if(items[i].items.total > 5) { %>
还有<%=items[i].items.total-5%>条评论 ) data-count=1 data-flag=true>点击查看
<%}%>
<%}%>
<%}%>

转载于:http://blog.itpub.net/7558084/viewspace-1002581/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值