第四章实现的仍然是一样的功能,只不过改为注解实现。
感觉注解是非常易用且方便的,但仍然觉得配置文件更好些,毕竟不用修改类文件。。。
依然是先来看目录结构:
从目录结构来看,与之前相比,最大的不同是Controller类由两个变为一个了,这也是注解的优势之一,可以使得一个类完成多个动作。
web.xml没有变化,不提了,先来看servlet的配置变化:
<?xml version="1.0" encoding="UTF-8"?>
<!--为了确保Spring能找到控制器,需在此添加该元素-->
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="
http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="app04a.controller"/><!--指定控制器类所在的package,越精确越好,避免不必要的扫描-->
<mvc:annotation-driven/><!--注册用于支持基于注解的控制器的请求处理方法的bean对象-->
<mvc:resources mapping="/css/**" location="/css/"/><!--指示对应的资源单独处理,不经过dispatcher servlet-->
<mvc:resources mapping="/*.html" location="/"/>
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"/>
<property name="suffix" value=".jsp"/>
</bean>
</beans>
变化最大的是Controller类,合二为一的新类为ProductController.java:
package app04a.controller;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import app04a.domain.Product;
import app04a.form.ProductForm;
@Controller // 表明这是一个控制器类
public class ProductController {
private static final Log logger = LogFactory.getLog(ProductController.class);
@RequestMapping(value = "/product_input") // 将对http://domain/product_input的访问映射到此方法,value为默认属性,故只配置value属性,则可以简写为@RequestMapping("/product_input"),如果该注解被用在控制器类上,则相当于URL多了一层,例如类被注释为"/PC",则访问该方法应访问"/PC/product_input",这点感觉相当灵活。
public String inputProduct() {
logger.info("inputProduct called");
return "ProductForm"; // 之前第三章的知识点,简化了路径,自动补全
}
@RequestMapping(value = "/product_save") // 这里的productForm实例是根据ProductForm.jsp所填写的内容自动生成的,规则是对应jsp中input标签的name属性。
public String saveProduct(ProductForm productForm, Model model) {
logger.info("saveProduct called");
// no need to create and instantiate a ProductForm
// create Product
Product product = new Product();
product.setName(productForm.getName());
product.setDescription(productForm.getDescription());
try {
product.setPrice(Float.parseFloat(productForm.getPrice()));
} catch (NumberFormatException e) {
}
// add product
model.addAttribute("product", product); // 无论是否使用,Spring MVC都会在每一个请求处理方法被调用时创建一个Model实例,用于增强需要显示在视图中的属性。这行执行之后,就可以在HttpServletRequest中访问了。
return "ProductDetails";
}
}
接下来4.4小节开始,介绍了@Autowired、@Service、重定向和Flash属性、请求参数和路径变量以及@ModelAttribute这几个知识点。
4.4小节,对ProductController.java改动较大,添加了新的私有变量productService,改变了save方法的逻辑内容,具体见代码:
ProductController.java:
package app04a.controller;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.servlet.mvc.support.RedirectAttributes;
import app04a.domain.Product;
import app04a.form.ProductForm;
import app04a.service.ProductService;
@Controller
public class ProductController {
private static final Log logger = LogFactory
.getLog(ProductController.class);
@Autowired // 会自动感知到@Service注解的类,此处为ProductServiceImpl
private ProductService productService;
@RequestMapping(value = "/product_input")
public String inputProduct() {
logger.info("inputProduct called");
return "ProductForm";
}
@RequestMapping(value = "/product_save", method = RequestMethod.POST)
public String saveProduct(ProductForm productForm, RedirectAttributes redirectAttributes) {
logger.info("saveProduct called");
// no need to create and instantiate a ProductForm
// create Product
Product product = new Product();
product.setName(productForm.getName());
product.setDescription(productForm.getDescription());
try {
product.setPrice(Float.parseFloat(productForm.getPrice()));
} catch (NumberFormatException e) {
}
// add product
Product savedProduct = productService.add(product); // 这是最大的变化,save方法不再是单纯跳转到详情页,而是先保存到了一个具体的类实例中
redirectAttributes.addFlashAttribute("message", "The product was successfully added."); // Spring3.1版本开始提供的Flash属性,可以实现重定向传值,需要在对应的方法上添加新的参数类型 org.springframework.web.servlet.mvc.support.RedirectAttributes
return "redirect:/product_view/" + savedProduct.getId();
}
@RequestMapping(value = "/product_view/{id}") // org.springframework.web.bind.annotation.RequestParam注释,获取请求参数并对应到同名的方法变量,使用路径变量有时会遇到由于默认上下文引起的错误,解决方案是使用JSTL标记的URL,例如@import url("<c:url value="/css/main.css"/>")
public String viewProduct(@PathVariable Long id, Model model) {
Product product = productService.get(id);
model.addAttribute("product", product);
return "ProductView";
}
}
ProductServiceImpl.java
package app04a.service;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.atomic.AtomicLong;
import org.springframework.stereotype.Service;
import app04a.domain.Product;
@Service // 与@Autowired搭配
public class ProductServiceImpl implements ProductService {
private Map<Long, Product> products = new HashMap<Long, Product>();
private AtomicLong generator = new AtomicLong();
public ProductServiceImpl() { // 启动web app时,该构造器会被调用,也就会添加id为1的product
Product product = new Product();
product.setName("JX1 Power Drill");
product.setDescription("Powerful hand drill, made to perfection");
product.setPrice(129.99F);
add(product);
}
@Override
public Product add(Product product) {
long newId = generator.incrementAndGet();
product.setId(newId);
products.put(newId, product);
return product;
}
@Override
public Product get(long id) {
return products.get(id);
}
}
springmvc-config.xml需要添加一行
<context:component-scan base-package="app04a.service"/>
为了避免使用路径变量造成错误,可以改用JSTL标记的URL
<%--<style type="text/css">@import url(/css/main.css);</style>--%>
<style type="text/css">@import url("<c:url value="/css/main.css"/>");</style><%--当程序部署为默认上下文时,链接标签会被转换为<style type="text/css">@import url(/css/main.css);</style>,上下文为context时,会转换为<style type="text/css">@import url(/context/css/main.css);</style>--%>