springMVC简介
是MVC框架,和Spring无缝连接。
第1讲:环境搭建
环境搭建都是从web.xml开始。
核心控制器DispacherServlet
说明:web.xml文件中,配置一个核心的servlet
1.在这之前要先拷贝jar包spring-framework-3.2.0.RELEASE,每一个特定jar包有三个(jar,文档,源码),真正的mvc包是spring-webmvc包。
框架有配置文件,springmvc框架就是springMVC-servlet.xml
springMVC-servlet.xml配置文件默认要放web-inf目录下。代码如下:
import org.springframework.web.servlet.mvc.Controller;
public class HelloWorldController implements Controller{
public ModelAndView handleRequest(HttpServletRequest req,
HttpServletResponse res)throws Exception{
return new ModelAndView("/welcome");
}
}
再建立一个welcome.jsp在根目录就可以做测试了。
启动报错,1.有可能是缺乏commons-logging.jar,
2.配置文件的名字错了,应为springMVC-servlet.xml
可以发请求测试了。localhost:8080/springMVC1/test1/helloworld
例子2:
public class HelloWorldController implements Controller{
public ModelAndView handleRequest(HttpServletRequest req,
HttpServletResponse res)throws Exception{
String hello = "1sh hello提高班";
return new ModelAndView("/welcome","result",hello);
}
}
页面是:${result}
例子3:
public class HelloWorldController implements Controller{
public ModelAndView handleRequest(HttpServletRequest req,HttpServletResponse res)throws Exception{
Map
map.put("map1","提高班1");
map.put("map2","提高班2");
return new ModelAndView("/welcome","map",map);
}
}要引入jstl的jar包和standard.jar包
页面是:
${m.key} --> ${m.value}
例子4:一个Controller多个方法
public class MultiController extends MultiActionController{
public ModelAndView add(HttpServletRequest req,HttpServletResponse res){
return new ModelAndView("/multi","method","add");
}
public ModelAndView update(HttpServletRequest req,HttpServletResponse res){
return new ModelAndView("/multi","method","update");
}
}
要在spring-servlet.xml文件里配置:
可以发请求测试了。
localhost:8080/springMVC2/test1/multi?action=add
localhost:8080/springMVC2/test1/multi?action=update
如果报错,可能是方法少了HttpServletRequest req,HttpServletResponse res
---------------------------------------------------------------------
图片的访问
1.在根目录建文件夹img,并把图片拷贝到这里。
2.jsp页面上:
例子5:
public class MultiController extends MultiActionController{
public ModelAndView img(HttpServletRequest req,HttpServletResponse res){
return new ModelAndView("/staticFile");
}
}
配置bean及配置静态资源访问(不拦截)
---------------------------------------------------------------------
使用注解方式的配置
springAnnotation-servlet.xml文件:
@Controller
public class UserController{
@RequestMapping(value="/user/addUser",method=RequestMethod.GET)
public ModelAndView addUser(){
String result = "this is addUser----";
return new ModelAndView("/annotation","result",result);
}
@RequestMapping(value="/user/delUser",method=RequestMethod.GET)
public ModelAndView delUser(){
String result = "this is delUser----";
return new ModelAndView("/annotation","result",result);
}
}
说明:按照上面的写法,请求 项目名/user/addUser会跳到annotation.jsp页面。
对于method=RequestMethod.GET或者POST,比较是严格的,不对应的请求是不会有效果的。
---------------------------------------------------------------------
注解的优化
1.开启注解可以用
2.当各个方法的请求url开头部分都相同的时候,可以合并在类上加注解。
类上RequestMapping(value="/user2"),方法就可以写RequestMapping(value="/addUser"),
相当 @RequestMapping(value="/user/addUser")
3.想把数据传递到页面可以用request.setAttribute("result",result);[要在方法的参数列表上加HttpServletRequest request]
4.get或者post方式不区分,可把 method=RequestMethod.GET 删掉
5.控制器返回值可以用字符串:return "/jquery";
前方式是:return new ModelAndView("/jquery","result",result);(返回页面及传参)
---------------------------------------------------------------------
参数传递
function addUser(){
var form = document.forms[0];//注意这里是forms
form.action="/springMVC6/user/data/addUser";
form.method="get";
form.submit();
}
@Controller
@RequestMapping("/user/data")
public class DataController {
@RequestMapping("/addUser")
public String addUser(String userName,String age,HttpServletRequest request){
request.setAttribute("userName",userName);
request.setAttribute("age",age);
return "/userManager";
}
}
说明:参数跟表单的name属性对应,就能接收到。
显示页面:
姓名是:${userName}
年龄是:${age}
说明:乱码是常见的。解决方法:server.xml里在
---------------第9讲---------------
$(function(){
$("#add").click(function(){
var userName = $("#userName").attr("value");
var age = $("#age").attr("value");
var user = {userName:userName,age:age};
$.ajax({
url:"/springMVC6/user/data/addUserJson",
type:"get",
data:encodeURI(user),
success:function(data){
alert("userName-->"+data.userName);
},
});
});
});
@RequestMapping("/addUserJson")
public String addUserJson(User user,HttpServletRequest request){
request.setAttribute("userName",user.getUserName());
request.setAttribute("age",user.getAge());
return "/userManager";
}
图片路径的注意点:
当类的路径配置是"/user/data"时,图片路径是../xxx的话,上一级目录就是user,不会返回到项目的根目录,
所以js引入路径最好是写src="/项目名/js/xxx"
---------------第10讲---------------
@RequestMapping("/addUserJson")
public String addUserJson(User user,HttpServletRequest request){
String result = "{\"userName\":\""+user.getUserName()+"\",\"age\":\""+user.getAge()+"\"}";
PrintWriter = null;
response.setContentType=("application/json");
try{
out = response.getWriter();
out.write(result);
}catch(IOException e){
e.printStackTrace();
}
}
拿json数据后台要out.write(),前台用jQuery的success回调函数就能拿结果function(data){}
---------------第11讲---------------
上传文件解析器配置
说明:上传enctype属性必须为multipart/form-data
少了两个上传文件的jar包,要拷贝。
另外文件上传
必须采用method="post"方式---------------第12讲:上传文件---------------
说的那个接口的方法用name属性的值
@RequestMapping("/upload2")
public void upload2(HttpServletRequest request,HttpServletResponse response){
CommonsMultipartResolver multipartResolver = new CommonsMultipartResolver(request.getSession().getServletContext());
if(multipartResolver.isMultipart(request)){
MultipartHttpServletRequest multiRequest = (MultipartHttpServletRequest)(request);
Iterator
while(iter.hasNext()){
MultipartFile file = multiRequest.getFile((String)iter.next());
if(file != null){
String fileName = "demoUpload" + file.getOriginalFilename();
String path = "D:/" + fileName;
File localFile = new File(path);
file.transferTo(localFile);
file.getInputStream();
}
}
}
}
尽量不用springmvc配置上传大小,在客户端验证,减少服务器压力。
---------------第13讲:与spring集成---------------
struts2和spring集成关键点:
1.struts.xml文件中,
说明:如果不用spring管理bean,要写包名+类名,有spring后写bean的id即可。
2.
没有声音
---------------第14讲:与spring集成2---------------
@Controller
public class SpringController{
@Resource(name="springManager")
private ISpring springManager;
@RequestMapping("/spring/get")
public String get(){
springManager.get();
return "/success";
}
}
---------------第15讲:与spring上下文关系---------------
//spring的上下文
WebApplicationContext ac1 = WebApplicationContextUtils.getWebApplicationContext(request.getSession().getServletContext());
//springMVC的上下文
WebApplicationContext ac2 = RequestContextUtils.getWebApplicationContext(request);
ISpring springManager = (ISpring)ac1.getBean("springManager");
在一个配置文件里写bean很臃肿,spring可以导入配置文件。
---------------第16讲:与hibernate简单实例---------------
- hibernate五大对象
Configuration
sessionFactory
Session
Transaction
Query和criteria
hibernate.cfg.xml里是这样:
public class User{
@Id
@GeneratedValue(generator="system-uuid")
@GenericGenerator(name="system-uuid",strategy="uuid")
private String id;
}
---------------第18讲:与hibernate简单实例---------------
需配置事务bean:
说明:事务管理哪个sessionFactory,可在里面指定
配置事务特性描述(策略):
---------------第19讲:整合简单实践---------------
public interface IUserDAO{
public void addUser(User user);
}
public class UserDAO implements IUserDAO{
public void addUser(User user){
}
}
另一教程
---------------------------------------------------------------------
核心原理
- 用户发送请求给服务器。
- 服务器接收到请求。发现DispacherServlet可以处理,于是调用DispacherServlet。
- DispacherServlet内部,通过HandleMapping用于检查一个url有没有对应的控制器。如果有,则调用,没则报404.
- 控制器开始执行。
- 控制器执行完毕后,如果返回字符串,则ViewResolver将字符串转化成相应的视图对象;
如果返回ModelAndView对象,该对象本身就包含了视图对象信息。
即改成return new ModelAndView("index");
- DispacherServlet将执行视图对象中的数据,输出给服务器。
- 服务器将数据输出给客户端。
注意:ModelAndView是web.servlet.ModelAndView包的
HandlerMapping
BeanNameUrlHandlerMapping
ModelAndView
-- 常用注解
- @Controller
在类上面加 @Component和 @Controller是等价的,表明是控制器。
- @RequestMapping
形式一:在类前面定义,则将url和类绑定。具体调哪个方法呢?需要url里边携带一个参数,如:
@RequestMapping(params="method=reg")
形式二:在方法前面定义,则将url和类的方法绑定。
- @RequestParam
比如有这样的一种情况,提交的表单的,
但是控制器的方法是public String reg3(String name){},
用于加在方法参数的前面,当提交的参数的name属性值跟参数不一致时,就可以加。
public String reg3( @RequestParam("uname") String name){//提交表单的name属性值是uname
....
}
说明:加 @RequestParam("uname")的意思就是把表单的属性值为uname所提交的数值赋值给name这个参数。
- @SessionAttributes({"u","a"})
说明:在某方法里如果想使用ModelMap,在方法的参数列表里加: ,ModelMap map 就可以(会自动注入);
如果想使用HttpServletRequest req,在方法的参数列表里加: ,HttpServletRequest req
从而req.getSession().setAttribute("x","xxx");
map.put("a","aaa");-->页面上${RequestScope.a}
在类上加了这个注解后,key为u和a也会在session里存一份。
- @ModelAttribute
这个注解可以跟 @SessionAttributes配合在一起用。可以将ModelMap中属性的值通过该注解自动赋给指定变量。
public String reg5( @ModelAttribute("u") String uname,ModelMap map){}
说明:这里有个uname参数,uname我不需要表单给我参数,我直接把ModelMap里的属性key为u的值直接给我赋过来。
请求转发与重定向
转发 return "forward:user.do?method=reg";不写也是转发
重定向 return "redirect:http://www.baidu.com";
modelAndView.setViewName("index");
User u = new User("jack");
modelAndView.addObject("uu",u);//较好用
return modelAndView;
页面上${requestScope.uu.uname}
class="org.springframework.web.servlet.view.InternalResourceViewResolver"> value="org.springframework.web.servlet.view.JstlView" />
/IMPM/WebContent/modules/saleManagement/earlyWarningManagement/earlyWarningManageDetail.jsp
-----------------------------------------------------------------
要让? @Autowired?起作用必须事先在?Spring?容器中声明AutowiredAnnotationBeanPostProcessor?Bean
?
当不能确定Spring容器中一定拥有某个类的bean时,可以用 @Autowired(required?=?false),这等于告诉?Spring:在找不到匹配?Bean?时也不报错
我们在?Spring?容器中配置了两个类型为?Office?类型的?Bean,当对?Boss?的?office?成员变量进行自动注入时,Spring?容器将无法确定到底要用哪一个?Bean,因此异常发生了。?Spring?允许我们通过?@Qualifier?注释指定注入?Bean?的名称,这样歧义就消除
了
@Resource作用相当于 @Autowired, @Resource默认按byName自动注入。如果使用name属性,则使用byName的自动注入策略;而使用type属性时则使用byType自动注入策略。
要让JSR-250的注释生效,除了加这些注释外,还需要在Spring容器中注册一个负责处理这些注释的BeanPostProcessor:
标注了 @PostConstruct的方法将在类实例化后调用,而标注了 @PreDestroy的方法将在类销毁之前调用。
@Component 注解的类就相当于xml配置文件里定义的bean。默认情况下通过 @Component定义的Bean都是singleton的,如果需要使用其它作用范围的Bean,可通过 @Scope 来达到目标。
==================第二套====================
- mvc在b/s系统下的应用,c控制器,m模型(pojo,action,service,dao),v视图
mvc是一个设计模式.
处理器映射器是干嘛的,是根据url找到对应的处理器。
处理器适配器去执行Handler。
第一步:发起请求到前端控制器DispatcherServlet
第二步:前端控制器查找Handler(可根据xml配置、注解进行查找)
第三步:处理器映射器HandlerMapping向前端控制器返回Handler(其实返回一个执行链)
第四步:前端控制器调用处理器适配器去执行Handler
第五步:处理器适配器去执行Handler
第六步:Handler执行完成,给适配器返回ModelAndView
第七步:处理器适配器向前端控制器返回ModelAndView(框架底层对象)
第八步:前端控制器请求视图解析器去进行视图解析。
第九步:视图解析器向前端控制器返回view
第十步:前端控制器进行视图渲染(将模型数据填充到request域中)
第十一步:前端控制器向用户响应结果
组件:
1.前端控制器DispatcherServlet
作用:接收请求,响应结果,相当于转发器。
2.处理器映射器
作用:根据请求的url查找Handler。
3.处理器适配器
作用:按照特定的规则去执行Handler
注意:编写Handler时按照HandlerAdapter的要求去做,这样适配器才可以去正确执行Handler。
4.视图解析器View resolver
作用:进行视图解析,根据逻辑视图名解析成真正的view
5.视图view
view是一个接口,实现类支持不同的view类型(jsp,freemarker,pdf...)
- 配置前端控制器(web.xml文件)
接下来要在springmvc.xml里配置 1.处理器映射器 2.处理器适配器 3.视图解析器
注意:多个映射器可以并存,前端控制器判断url能让哪些映射器映射,就让正确的映射器处理。
这时java代码是:
public class ItemsController implements Controller
request.setAttribute("itemsList",itemsList);
//视图
request.getRequestDispatcher("/WEB-INF/jsp/items/itemsList.jsp").forward(request,response);
好处是可以响应json数据
===================分隔线===================
注意点3.多个映射器可以并存,前端控制器判断url能让哪些映射器映射,就让正确的映射器处理。
非注解的处理器适配器
org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter
要求编写的Handler实现Controller接口
org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter
要求编写的Handler实现HttpRequestHandler接口。
映射器
在spring3.1之前使用org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping注解映射器。
在spring3.1之后使用org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping注解映射器。
适配器
在spring3.1之前使用org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter注解适配器。
在spring3.1之后使用org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter注解适配器。
public class ItemController implements Controller{
public ModelAndView handleRequest(HttpServletRequest request,HttpServletResponse response) throws Exception{
//调用service
List
//填充静态数据,模拟数据库
Items items_1 = new Items();
items_1.setName("");
ModelAndView modelAndView = new ModelAndView();
//相当于request的setAttribute,在jsp页面可取数据
modelAndView.addObject("itemsList",itemsList);
//指定视图
modelAndView.setViewName("/WEB-INF/jsp/items/itemsList.jsp");
return modelAndView;
}
}
小结:
重点掌握注解的方式。
对标记有@Controller的类中标记有@RequestMapping的方法进行映射。
视图解析器
表现层 |
业务层 |-- spring将各层进行整合、事务控制。
持久层 |
第一步:整合dao层。
使用mapper的扫描器自动扫描mapper接口,在spring中进行注册。
第二步:整合service层。
将service接口配置在spring配置文件中。
第三步:
所需要的jar包:
数据库驱动
log4j包
数据库连接池包
jstl包
spring
items.name LIKE '%${itemsCustom.name}%'
SELECT items.* FROM items
/**商品包装对象*/
public class ItemsQueryVo{
private Items items;//商品信息
private ItemsCustom itemsCustom;
}
/**商品信息的扩展*/
public class ItemsCustom extends Items {
//添加扩展属性
}
public class Items{
这里是自动生成的
}
接口
public interface ItemsMapperCustom{
public List
}
21小节:
url映射
窄化请求路径
限制请求,配置的是允许的类型。
24小节:
参数绑定默认支持的类型
HttpServletRequest
HttpServletResponse
HttpSession
Model/ModelMap model是接口,ModelMap是一个接口实现
简单类型 Integer
pojo
自定义转换器
跟struts2区别:
1.springmvc将url和controller方法映射,映射成功后springmvc生成一个Handler对象,对象中只包括了一个method,方法执行结束,形参数据销毁。
2.springmvc可以进行单例开发,并建议使用单例开发,struts2通过类的成员变量接收参数,无法使用单例,只能使用多例。
3.struts2速度慢,在于使用struts标签,建议使用jstl
springmvc第二天
复习
DispatcherServlet 前端控制器:接收request,进行response
接收到请求之后,需要调用一个组件HandlerMapping处理器映射器:根据url查找Handler。(xml配置或者注解)
HandlerAdapter处理器适配器:根据特定规则去执行Handler,编写Handler时需要按照HandlerAdapter的要求去编写。
Handler处理器(后端控制器):需要程序员去编写,常用注解开发方式。
Handler处理器执行后结果 是ModelAndView(Adapter处理决定的),具体开发时Handler返回方法值类型包括:ModelAndView、
String(逻辑视图名)、void(通过在Handler形参中添加request和response,类似servlet开发,可实现json数据输出)
View resolver视图解析器:根据逻辑视图名生成真正的视图
View视图:jsp页面,仅是数据展示,没有业务逻辑。
自定义参数绑定
定义的Converter<源类型,目标类型>接口
注意:要转换的目标类型一定和接收的pojo中的属性类型一致。
将定义的Converter实现类注入到处理器适配器中。
学习内容
绑定List
进入批量修改页面,批量修改的提交
使用List接收页面提交的批量数据,要通过包装类定义List
绑定Map:同样通过包装类,页面也是定义属性名,info['name']
- validation校验
@Size(min=1,max=30,message="{items.name.length.error}",groups={ValidGroup1.class}) 校验组1
private String name;
@NotNull(message="{items.createtime.isNull}") 非空校验
private Date createtime;
ValidationMessages.properties文件:
items.name.length.error=请输入1到30个字符
items.createtime.isNull=请输入生成日期
public String editItemsSubmit(Model model,@Validated(value={ValidGroup1.class}) ItemsCustom itemsCustom,BindingResult bindingResult)throws Exception{
if(bindingResult.hasErrors()){
//输出错误信息
List
for(ObjectError objectError:allErrors){
System.out.println(objectError.getDefaultMessage());
}
model.addAttribute("allErrors",allErrors);
}
}
页面上:
${error.defaultMessage}
- 分组校验
校验分组接口
public interface ValidGroup1{
}
public interface ValidGroup2{
}
- 数据回显
@ModelAttribute("items") 可以指定pojo回显到页面在request中的key
- JSON交互
1.要求请求的是json串,所以在前端页面中需要将请求内容转成json,不方便。
2.请求key/value,输出json,常用。