因为比较感兴趣,所以今天学习了一下springmvc。学习过程中可能还有不对的地方,还希望大家多指正批评。
spring用的是3.2.2,对数控据库的操作用的是jpa和hibernate4,重点介绍一下我学习springmvc的过程和中间遇到的问题;
根据spring的文档,在web.xml配置DispatcherServle
<servlet>
<servlet-name>example</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>example</servlet-name>
<url-pattern>/app/*</url-pattern>
</servlet-mapping>
然后需要在WEB-INF配置一个文件,名字结构为 [servlet-name]-servlet.xml
,那么按照上边servlet的配置,这个文件的名字叫做"example-servlet.xml"。如果我们不想把文件放在WEB-INF下的话,可以通过配置参数,例如
servlet>
<servlet-name>example</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:com/peng/springmvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
这样来指定配置文件的位置,当然,名字也可以更换,不必按之前的命名规范。至于这个文件的内容,
和spring配置文件的结构一样。另外需要配置以下几项:
1.<mvc:annotation-driven /> 对spring mvc提供注解支持;
2.<context:component-scan base-package="com.base" />设置扫描的包;
3.设置Resolver
<bean id="viewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/" />
<property name="suffix" value=".jsp" />
</bean>
接下来编写Controller,
@Controller
public class MyController {
@RequestMapping(value = "/exam/{id}",method = RequestMethod.GET)
public String test(@PathVariable("id") int id,@RequestParam("name") String name,Model model){
System.out.println("request exucte");
System.out.println("pathVariable id = "+id);
System.out.println("param name = "+name);
return "success";
}
}
对Controller,
1.需要在类上加@Controller注解,以便让spring容器扫描,以bean的形式加入到容器中。
2.在方法上,需加上@RequestMapping注解,根据注解的意思不难理解,就是用来映射请求uri。value的值都是绝对值,不是相对值,这点要注意。而且不需要在web.xml的url-pattern中配置的路径。例如在web.xml的url-pattern中配置的路径"/app/*",那么对于这个请求“http://localhost:8080/testssh/app/exam/index”,@RequestMapping()中的value="/exam/index",即@RequestMapping(value="/exam/index")。对于method,则是和请求想对应,请求是GET,则对应的method= RequestMethod.GET,post同理。
Controller类上也可以加@RequestMapping注解,如果这个类的方法都是用来处理“http://localhost:8080/testssh/app/exam“下的请求,那么类上的@RequestMapping可以写成@RequestMapping(value="/exam");
3.如果想得到请求路径上的某个值,可以通过在@RequestMapping的value上用"{}"指定,然后在方法参数前,通过@PathVariable来得到这个值。请求参数的话通过@RequestParam来获得,正如上面的例子一样。
4.返回值有多种,我这里用的是最简单的一种,即返回一个jsp页面的名称。如果上面的test方法执行完,那么就会转到WEB-INF/jsp/success.jsp页面。这里你应该也明白了配置Resolver的作用了吧。
上边只是最简单的一些应用,能达到有请求有响应。
下面做一个保存的例子,entity类和service层都已经做好,只需要把前台输入的数据传入到Controller,然后调用service层的save方法保存就好了。
TestEntity类
@Entity
@Table(name="testentity")
public class TestEntity {
private int id;
private String name;
private boolean sex;
private int age;
private Date birthday;
private double height;
@Id
@GeneratedValue
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public boolean isSex() {
return sex;
}
public void setSex(boolean sex) {
this.sex = sex;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Temporal(TemporalType.DATE)
public Date getBirthday() {
return birthday;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
@Type(type="double")
public double getHeight() {
return height;
}
public void setHeight(double height) {
this.height = height;
}
}
之所以把entity类列出来,主要是为了说明date类型,无论struts2还是spring mvc都会遇到时间类型的解析问题,这个下面会说道。
一.对于前台页面的form,我用的是spring对表单提供的标签。
1.首先引入:<%@taglib uri="http://www.springframework.org/tags/form" prefix="form" %>
2.然后编写form.我编写的form格式如下:
<form:form action="exam/save" method="post" commandName="test">
名称:<form:input path="name"/><br/>
性别:<form:radiobutton path="sex" value="true" label="男"/>
<form:radiobutton path="sex" value="false" label="女"/>
<br/>
年龄:<form:input path="age" title="年龄" /><br/>
生日:<form:input path="birthday" title="生日"/><br/>
身高:<form:input path="height" title="身高"/><br/>
<input value="Submit" type="submit">
</form:form>
这里涉及到一个表单对象,这里指的是Entity。这个entity放在pageContext中,默认情况下,表单控制器将表单对象以“command”为名放到PageContext中,你可以通过表单控制器commandName属性的设置使用其它的名字(假设设置为“test”),这时你必须通过<form:form commandName="test">显式指定绑定的表单对象名称。当你访问这个页面的时候,PageContext中必须包含这个表单对象,例如commandName="test",如果不包含,则会报"either BindingResult nor plain target object for bean name 'test' available as request attribute"错误。所以首先需要在pageContext中放入这个表单对象。
@RequestMapping(value="/index")
public ModelAndView index(Model model){
TestEntity test = new TestEntity();
model.addAttribute("test", test);
return new ModelAndView("form", model.asMap());
}
通过
访问一个请求,将表单对象放到model中,然后返回ModelAndView,其中的参数"form"指定form页面的名称,spring会转到form.jsp页面。这样就可以访问了。
然后写一个方法用于接收表单数据,并处理表单数据;
@RequestMapping(value= "/exam/save",method=RequestMethod.POST)
public String save( TestEntity test,Model model){
getTestService().save(test);//
return "success";
}
然后填写数据就可以提交了,等待奇迹的出现。但出现的不是奇迹,出现的是400错误。这并不让人惊讶,因为每个开发人员都有一颗异常蛋定的心。看看是为什么,开始以为是form标签错误,但检查来检查去,并没有发现异常。然后通过浏览器的开发者工具,发现提交的数据也正常,那么发生错误的应该是服务器没有接受数据,这也正是400代表的意思啊。请求地址是对的,为啥没解析组装成entity对象呢?百度,google了好一阵,没有发现问题在哪,很郁闷!忽然看到关于date解析造成400的搜索结果,赶紧去看了下,意思是要注册web数据的编辑器,就是要对特殊数据进行转换,下面这段代码同样写到Controller类中。
@InitBinder
public void initBinder(WebDataBinder binder) {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
dateFormat.setLenient(false);
binder.registerCustomEditor(Date.class, new CustomDateEditor(dateFormat,false));
}
当然,注册的方式很多,我只是用了一种。大家可以在spring的文档中看一下。
注册完之后,提交数据,ok!
对于前台form,也可以用最普通的方式,提交的效果是一样的。spring提供的form标签的好处在于能够回显等方面。
以上是我初步学习的总结,很显然,这仅仅是学习如何让使用这个工具,但对于工具原理的理解还不多。想要更好地利用工具,必须更好地理解工具,这样才能用起来更加灵活和方便!