springMVC个人笔记

springMVC是采用MVC设计模式的一个表现层轻量级web框架,是最主流的MVC框架之一
springMVC中有Model模型 View视图 Controller控制器以及springMVC特有的Front Controller前端控制器

环境搭建

创建maven工程,选择骨架org.apache.maven.archetypes:maven-archetype-webapp
补充目录结构java,resources,test
导入spring-context,spring-web,wpring-webmvc以及jsp,servlet的jar包

在controller包类(类似于web),在这个包下创建一个类

@Controller                                                     将该类放入spring容器中
//@RequestMapping(path="/users")                                可以使以下所有方法的@RequestMapping省略/users(已注释)
public class helloController { 

    @RequestMapping(path="/hello")                              请求路径
    public String sayHello() {
        System.out.println("你好springMVC");
        return "success";                                       跳转到success.jsp
    }
}

在resources下创建一个springmvc.xml

<?xml version="1.0" encoding="UTF-8"?>                         需要spring-beans,mvc,context的约束
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:mvc="http://www.springframework.org/schema/mvc"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       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="cn.le"/>                  开启包扫描
<bean id="InternalResourceView" class="org.springframework.web.servlet.view.InternalResourceViewResolver">创建视图解析器
   <property name="prefix" value="/WEB-INF/pages/"></property>  配置前缀(除了index.jsp所有页面都放在pages目录下)
   <property name="suffix" value=".jsp"/>                       配置后缀
</bean>
<mvc:annotation-driven/> 
</beans>

配置webapp下的web.xml

<?xml version="1.0" encoding="UTF-8"?>                           web.xml需要的约束(Servlet3.1)
<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">
<display-name>springmvc-quick</display-name>                    命名(描述性文字,无实际作用)
<welcome-file-list>
   <welcome-file>index.jsp</welcome-file>                       配置初始页
</welcome-file-list> 
<servlet>
  <servlet-name>springmvc</servlet-name>                        核心控制器
  <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
  <init-param>                                                  配置初始化参数
       <param-name>contextConfigLocation</param-name>
       <param-value>classpath:springmvc.xml</param-value>       通过这个参数才能让springmvc.xml被加载
   </init-param>
   <load-on-startup>1</load-on-startup>                         启动项(在服务器启动的时候就开始初始化)
</servlet>
<servlet-mapping>
    <servlet-name>springmvc</servlet-name>
    <url-pattern>/</url-pattern>                                拦截请求(/* 匹配所有资源 /匹配除了jsp以外的所以资源)
</servlet-mapping>
</web-app>

运行流程
先创建核心容器->springmvc.xml被加载->扫描注解->将controller放入spring容器中
                                                                                 ->创建视图解析器
                                                                                 ->打开springmvc注解的支持

当有资源访问的时候,视图解析器通过访问路径调度到对应的方法上(通过@RequestMapping),最后通过return的值跳转到对应的页面上

备注
@RequestMapping这个注解类似于底下代码


public class BaseServlet extends HttpServlet {                               继承HttpServlet,重写service方法
    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        String uri = req.getRequestURI();                                    获取请求路径
        String methodName = uri.substring(uri.lastIndexOf('/') + 1);         获取方法名(通过截取请求路径获取)
        try {
            Method method = this.getClass().getMethod(methodName, HttpServletRequest.class, HttpServletResponse.class);
            Object invoke = method.invoke(this, req, resp);                  反射相关操作
        } catch (NoSuchMethodException e) {
            e.printStackTrace();
        } catch (IllegalAccessException e) {
            e.printStackTrace();
        } catch (InvocationTargetException e) {
            e.printStackTrace();
        }
    }
}

代码解释
Servlet类都是一个功能对应一个Servlet,通过抽取方法将一堆功能继承在一个Servlet,增加代码可读性维护性
将需要调用的Servlet继承BaseServlet就能实现,并写好注解如@WebServlet("/user/*")
注解这个时候无实际意义,只是使用反射调用对应方法的标记而已
访问/user/add->this(谁调用谁就是this)通过字符串截取获取add->通过反射调用user下的add方法

@RequestMapping注解详解

@RequestMapping(value="/hello",method = RequestMethod.POST) 这个方法只有是Post请求才会被调用(超链接是get请求)
@RequestMapping(value="/hello",params = {“username”}) 接受的参数里必须有username
@RequestMapping(value="/hello",params = {“username=nm$”}) 接收的参数的键和值都要相同
@RequestMapping(value="/hello",headers = {“Accept”}) 请求头里必须有Accept这个参数

springMVC的自动赋值

准备HTML页面

<form action="hello7" method="get">                              
账号<input type="text" name="username"><br/>                     为account里的元素赋值
密码<input type="text" name="password"><br/>
ID<input type="text" name="id"><br/>
用户2<input type="text" name="user.username"><br/>               为account里的user这个实体类里的元素赋值
ID2<input type="text" name="user.id"><br/>
<input type="submit" value="登入!">
</form>

必须有成员变量以及对应的set方法
*获取方法为setUsername->Username->username

@RequestMapping(value="/hello")
public String sayHello(String username,String password) {       如果请求接受到username和password,springMVC将自动为方法参数赋值
     return "success";
}

@RequestMapping(value="/hello")
public String sayHello2(Account account) {                       如果是一个javaBean对象,将自动封装数据                
     return "success";
    }

为集合赋值
account里再提供List<User  >,Map<String,User>以及对应的set方法

用户2<input type="text" name="list[0].username"><br/>            为List集合赋值
ID2<input type="text" name="list[0].id"><br/>
 
用户3<input type="text" name="map['first'].username"><br/>       为Map集合赋值
ID4<input type="text" name="map['first'].id"><br/>

中文问题

在web.xml中配置中配置过滤器

<filter>
    <filter-name>filter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
         <param-name>encoding</param-name>
         <param-value>UTF-8</param-value>                        编码为UTF-8
    </init-param>
</filter>
<filter-mapping>
    <filter-name>filter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

选择不拦截的路径

<mvc:resources mapping="/js/" location="/js/**"/>
<mvc:resources mapping="/images/" location="/js/**"/>
<mvc:resources mapping="/css/" location="/css/**"/>      这些资源不拦截

自定义类型转换器

必须实现Converter<S,T>并重写convert方法

public class StringToDate implements Converter<String, Date> {

    @Override
    public Date convert(String source) {
        if (source == null) {
            System.err.println("参数不能为空");
            throw new RuntimeException();
        }
        Date parse = null;
        try {
            parse = new SimpleDateFormat("yyyy-MM-dd").parse(source);
        } catch (ParseException e) {
            System.err.println("输入的参数无法转换");
            throw new RuntimeException();
        }
        return parse;
    }
}

配置自定义类型转换器

 <!--自定义类型转换器-->
<bean id="conversionService" class="org.springframework.context.support.ConversionServiceFactoryBean">   
        <property name="converters">
            <set>
                <bean class="cn.le.Converter.StringToDate"></bean>
            </set>
        </property>
    </bean>
 <!--springMVC注解的支持,将自己的代码也生效的配置-->
 <mvc:annotation-driven conversion-service="conversionService"/>

springMVC的特殊操作

获取Servlet的原生对象

public String sayHello(HttpServletRequest req, HttpServletRequest res) {  直接放在参数里,就能调用

参数不一致导致无法封装解决

@RequestMapping(value = "/hello11")
public String sayHello11(@RequestParam("name=username")String name) {     如果前端参数不一致加这个注解
@RequestMapping(value = "/hello11")
public String sayHello11(@RequestParam(required=true) String name) {      必须和方法里的参数一致,不然报错

获取整个请求体

@RequestMapping(value = "/hello12")                                       这个是请求体:user=321&age=312
public String sayHello12(@RequestBody String body) {                      通过这个注解直接获取后端发送的完整请求体(get不行)

使用restful风格

@RequestMapping("/hello13/{uid}")                                         前端的href为href="hello13/10,获取到uid为10,并赋值给id
public String sayHello13(@PathVariable(name ="uid") String id) {          @PathVariabl注解通过name找到对应{}内的值

获取请求头

@RequestMapping("/hello14")
public String sayHello14(@RequestHeader("Accept") String Accept) {        通过这个注解获取请求头Accept

获取指定cookie

@RequestMapping("/hello15")
public String sayHello15(@CookieValue("JSESSIONID") String JSESSIONID) { 通过这个注解获取指定名称的cookie

@ModelAttribute注解

有返回值用法
@RequestMapping("/hello16")                      
public String sayHello16(User user) {                接收到下面方法的user
  return "success";
 }
@ModelAttribute                                      即使没调用这个方法,有了@ModelAttribute注解这个方法总是会执行,并且先执行                                 
public User say(String uname) {                      一个表单可能无法为实体类的所有元素赋值,先从数据库里查出来并赋值,然后在调用hello6将其中的部分参数覆盖         
User user=(数据库执行操作)                                               
   return user;
    }
无返回值用法
@RequestMapping("/hello16")
public String sayHello17(@ModelAttribute("abc") User user) { 从下面的Map集合中取
    System.out.println("我执行sayHello");
    return "success";
    }
@ModelAttribute
public void say2(String username, Map<String, User> map) {  将user存入Map集合中
    System.out.println("我执行说");
    User user=(数据库执行操作)
        map.put("abc", user);
    }

共享参数

@SessionAttributes(value = "msg")                         再在类上写上这个注解代表将从request与中取出再存入session域中
public class helloController {
@RequestMapping("/hello18")
public String sayHello18(Model model) {                   先存入Model里的键值对会被springMVC存入request域中
     System.out.println("我执行sayHello");
     model.addAttribute("msg", "123");
     return "success";
    }

控制器无返回值用法

@RequestMapping("/test")
public void test() {                                     如果是无返回值方法那么会跳转到请求路径上                       
    System.out.println("进入了");<a href="test">点击</a>会跳转到test这个路径下
                                                         当然也可以使用HttpServlet原生API来进行跳转(跳过视图解析器,自己写代码)

不使用视图解析器转发

方法返回值写成这样就行了
 return "forward:/WEB-INF/pages/success.jsp";            转发写法
 return "redirect:/index.jsp";                           重定向写法

ModelAndView

@RequestMapping("/test3")
public ModelAndView test3(){
ModelAndView mv=new ModelAndView();
mv.addObject("user",user);                               用ModelAndView存入request域中
mv.setViewName("success");                               用ModelAndView跳转指定页面
    return mv;

获取ajax请求(需要jackson的jar包)

直接获取用@RequestBody
@RequestMapping("/testAjax")
public @ResponseBody User testAjax(@RequestBody User user){ @ResponseBody可以写在返回值类型里也可以写在参数里
                                                            springMVC会自动转换为jackson或ajax请求

*返回值如果是字符串为地址,是实体类为json数据

文件上传

准备HTML页面

<a href="response.jsp">啦啦啦</a>
<form action="" method="post" enctype="multipart/form-data">      enctype的值使form表单能传递文件
    选择文件<input type="file" name="upload"/>
    <input type="submit" value="上传文件"/>
</form>

不使用springMVC(需要commons-fileupload和commons-io的jar包)

  @RequestMapping("/file")
    public String testFile(HttpServletRequest req) {
        //通过req获取session,通过session获取最大域对象context,然后获取指定路径
        String path = req.getSession().getServletContext().getRealPath("/uploads");
        File file = new File(path);
        //判断是否存在
        if (!file.exists()) {
            file.mkdirs();
        }
        //获取commons-fileupload的工厂类
        DiskFileItemFactory factory = new DiskFileItemFactory();
        //获取实现上传的类
        ServletFileUpload upload = new ServletFileUpload(factory);
        List<FileItem> fileItems = null;
        try {
            //解析req,获取文件项
            fileItems = upload.parseRequest(req);
        } catch (FileUploadException e) {
            e.printStackTrace();
        }
        //遍历
        for (FileItem f : fileItems) {
            //如果不是普通表单,那就是上传文件的表单
            if (!f.isFormField()) {
                String name = f.getName();
                try {
                    f.write(new File(path, name));
                    //删除临时文件
                    f.delete();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }
        return "success";

    }

springMVC的上传方式
使用MultipartFile

 @RequestMapping("/springMVC")
    public String testFile2(HttpServletRequest req, MultipartFile upload) { //必须要和表单的name属性一致
        //通过req获取session,通过session获取最大域对象context,然后获取指定路径
        String path = req.getSession().getServletContext().getRealPath("/uploads");
        File file = new File(path);
        //判断是否存在
        if (!file.exists()) {
            file.mkdirs();
        }
        String fileName = upload.getOriginalFilename();
        try {
            //移动到新的地方
            upload.transferTo(new File(path, fileName));
        } catch (IOException e) {
            e.printStackTrace();
        }

        return "success";

    }

配置springMVC

 <!--配置文件解析器对象-->
    <bean id="multipartResolver" class="org.springframework.web.multipart.commons.CommonsMultipartResolver">
        <property name="maxInMemorySize" value="10485760"/>
    </bean>
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值