IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 开发工具 -> SpringMVC02_idea创建Web项目的方式_Url的Ant风格_Rest风格_MVC处理请求数据_【请求响应乱码问题】_MVC处理响应数据 -> 正文阅读

[开发工具]SpringMVC02_idea创建Web项目的方式_Url的Ant风格_Rest风格_MVC处理请求数据_【请求响应乱码问题】_MVC处理响应数据

idea创建Web项目的三种方式

方式一:

在idea中创建web项目【便于打war包,让实施人员部署程序工作,推荐使用】

1.新建空的maven项,
2.在main文件夹中添加webapp文件夹,再在webapp文件夹下创建WEB-INF文件夹
3.在WEB-INF文件夹下创建web.xml
4.在project Settings中的Facets项目中添加Web项目
5.在project Settings中的Modules项目中修改Deployment Descriptors的路径为web.xml的路径,修改Web Resource Directories的路径为webapp的路径
注意查看project Settings中的Artifacts项目中WEB-INF/classes下有没有lib目录看看有没有jar包

方式二:

  1. 创建一个空的maven项目【不使用idea自带的骨架】
  2. 右键项目添加web的支持 Add Frameworks Support

方式三:
创建一个mavean项目勾选idea自带的骨架【org.apache.maven.archetypes:maven-archetype-webapp】

Ant风格URL

  • Ant 风格资源地址支持 3 种匹配符

    a) ?:匹配一个字符

    b) *:匹配任意字符

    c) **:匹配任意层、任意字符路径

  @RequestMapping("/successful/**/?")
public String testHelloWorld(){
    return "successful";
}

在这里插入图片描述

@PathVariable注解

  • 作用:将占位符中参数,注入处理请求方法的形参中,要求参数名一致。
  • 属性
    • value:设置入参参数名 如果只有value属性时,value可以省略不写
    • name:与value作用一致
    • required:设置当前参数是否必须入参
      • true:必须入参,如不入参会报错:500
      • false:不必须入参,如不入参不会报错,会将null值注入
   @RequestMapping("/successful/{id}") //{id}获取浏览器URL中的请求参数 必须和下边方法中的形参名称一致
    public String testHello(@PathVariable(value = "id") Integer id){
        System.out.println("id = " + id);
        return "successful";
    }

REST简介

1 REST与传统URL风格对比

  • 传统CRUD的URL风格
    • 业务 URL 请求方式
    • 增 /addUser POST
    • 删 /deleteUser?id=1001 GET
    • 改 /updateUser POST
    • 查 /getUser?id=1001 GET
  • REST_CRUD的URL风格
    • 业务 URL 请求方式
    • 增 /user POST
    • 删 /user/1001 DELETE
    • 改 /user PUT
    • 查 /user/1001 GET
  • 总结:REST风格CRUD优势
    • 安全!!!因为URL都一样 不知道参数名称 可以防止侵入 传统的URL可以看到参数名称
    • 提高网站排名
    • 便于第三方平台对接
    • HTTP协议底层支持,效率高

2 HiddenHttpMethodFilter源码解析

解决遇到新问题正确思路:第一方【JDK】、第三方、第二方【自己】

  • public static final String DEFAULT_METHOD_PARAM = "_method";
    private String methodParam = DEFAULT_METHOD_PARAM;
    
    @Override
    protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
          throws ServletException, IOException {
    
       HttpServletRequest requestToUse = request;
    
       if ("POST".equals(request.getMethod()) && request.getAttribute(WebUtils.ERROR_EXCEPTION_ATTRIBUTE) == null) {
          String paramValue = request.getParameter(this.methodParam);
          if (StringUtils.hasLength(paramValue)) {
             String method = paramValue.toUpperCase(Locale.ENGLISH);
             if (ALLOWED_METHODS.contains(method)) {
                requestToUse = new HttpMethodRequestWrapper(request, method);
             }
          }
       }
    
       filterChain.doFilter(requestToUse, response);
    }
    /**
    	 * Simple {@link HttpServletRequest} wrapper that returns the supplied method for
    	 * {@link HttpServletRequest#getMethod()}.
    	 */
    	private static class HttpMethodRequestWrapper extends HttpServletRequestWrapper {
    
    		private final String method;
    
    		public HttpMethodRequestWrapper(HttpServletRequest request, String method) {
    			super(request);
    			this.method = method;
    		}
    
    		@Override
    		public String getMethod() {
    			return this.method;
    		}
    	}
    
  • 总结:使用HiddenHttpMethodFilter需要准备如下两步

    - 提交方式必须是:POST
    - 必须提交请求参数:_method

3 HiddenHttpMethodFilter基本使用

  • xml

    <!--    注册HiddenHttpMethodFilter,为REST风格准备,配置HiddenHttpMethodFilte过滤器,目的是为了将POST请求转换为PUT或DELETE请求-->
    <filter>
        <filter-name>HiddenHttpMethodFilter</filter-name>
        <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
    </filter>
    <filter-mapping>
        <filter-name>HiddenHttpMethodFilter</filter-name>
        <url-pattern>/*</url-pattern>
    </filter-mapping>
    
  • html

    <h2>修改user信息</h2>
    <form th:action="@{/RestController}" method="POST">
        <input type="hidden" name="_method" value="PUT">
        <input type="submit">
    </form>
    <h2>删除user信息</h2>
    <form th:action="@{/RestController}" method="POST">
        <input type="hidden" name="_method" value="DELETE">
        <input type="submit">
    </form>
    
  • java

    /**
     * 修改user信息
     * @return
     */
    @RequestMapping(value = "/RestController",method = RequestMethod.PUT)
    public String updateUser(){
        System.out.println("修改User信息...");
        return "success";
    }
    
    /**
     * 删除user信息
     *
     * @return
     */
    @RequestMapping(value = "/RestController",method = RequestMethod.DELETE)
    public String deleteUser(){
        System.out.println("删除User信息...");
        return "success";
    }
    

SpringMVC处理请求数据

JavaWeb阶段【Servlet中处理请求数据-request】

  • 获取请求参数
    • request.getParameter()
    • request.getParameterValues()
    • request.getParameterMap()
  • 获取请求头
    • request.getHeader()
  • 获取请求行【请求方式、uri】
    • request.getMethod()
    • request.getRequestURI()
    • request.getRequestURL()
  • request.getCookies():获取cookie信息

1 SpringMVC获取请求参数【重要】

  • 默认情况:当参数名与处理请求方法中形参名一致时,SpringMVC可以实现自动入参 即可以不加@RequestParam注解
  • @RequestParam注解【重要】
    • value:设置入参参数名
    • name:与value属性作用一致
    • required:设置当前参数,是否必须入参
      • true【默认值】:必须入参,如未入参会报错:400
      • false:不必须入参,如未入参,会将null值注入【建议使用包装类】
    • defaultValue:设置默认值
      • 只有未入参,就会为指定参数设置默认值
<a th:href="@{/successful999(id=1001,name='秘书部')}">点击访问successful页面@RequestParam</a><br>
  @RequestMapping("/successful999")
  public String testHello3(@RequestParam(value ="id", required = true) Integer deptId,@RequestParam(value ="name",
          required = true)String deptName){
      System.out.println("deptId = " + deptId);
      System.out.println("deptName = " + deptName);

      return "successful";
  }

2 SpringMVC使用POJO入参【重要】

  • 必须要保证请求参数名与POJO的属性名保持一致。支持级联属性入参

  • 示例

    <h2>添加员工信息</h2>
    <form th:action="@{/addEmp}" method="post">
        员工id:<input type="text" name="empId"><br>
        员工姓名:<input type="text" name="empName"><br>
        员工薪资:<input type="text" name="salary"><br>
        员工部门ID:<input type="text" name="dept.id"><br>
        员工部门名称:<input type="text" name="dept.deptName"><br>
        <input type="submit" value="提交员工">
    </form>
    
    @RequestMapping("/addEmp")
    public String addEmp(Emp emp){
        System.out.println("emp = " + emp);
        return "success";
    }
    
    public class Emp {
        private Integer empId;
        private String empName;
        private Double salary;
        private Dept dept;
        //....
    }
    public class Dept {
        private Integer id;
        private String deptName;
    	//...
    }
    

3 SpringMVC获取请求头

  • @RequestHeader:获取请求头

    • 属性

      • value:设置获取请求头参数名称
      • name:与value作用一致
      • required:设置当前请求头是否必须入参
        • true【默认值】:必须入参
        • false:不必须入参
      • defaultValue:设置默认值
        • 如有未入参请求头,则设置当前默认值
    • 示例代码

      @RequestMapping("/getRequestHeader")
      public String getHeader(@RequestHeader("User-Agent")String uAgent,
                              @RequestHeader("Referer")String referer,
                              @RequestHeader(value = "Cookie",required = false)String cookie){
          System.out.println("uAgent = " + uAgent);
          System.out.println("referer = " + referer);
          System.out.println("cookie = " + cookie);
          return "success";
      }
      

4 SpringMVC获取CookieValue

  • @CookieValue

    • 属性
      • value:设置获取CookieValue参数名称
      • name:与value作用一致
      • required:设置当前CookieValue是否必须入参
        • true【默认值】:必须入参
        • false:不必须入参
      • defaultValue:设置默认值
        • 如有未入参CookieValue,则设置当前默认值
  • 示例代码

    @RequestMapping("/getCookieValue")
    public String getCookieValue(HttpServletRequest request, @CookieValue(value = "JSESSIONID",required = false)String cValue){
        HttpSession session = request.getSession();
        System.out.println("sessionId:" + cValue);
        return "success";
    }
    

5 SpringMVC中支持原生Servlet中API

  • 如需要原生Servlet底层API,直接将对象入参即可
@RequestMapping("/getCookieValue")
    public String getCookieValue(HttpServletRequest request,
                                HttpSession session,
                                 @CookieValue(value = "JSESSIONID",required = false)String cValue){
//        HttpSession session = request.getSession();
//        System.out.println("sessionId:" + cValue);
        String id = session.getId();
        System.out.println("id = " + id);
        return "success";
    }

SpringMVC中解决【请求与响应】乱码问题

JavaWeb中如何解决乱码问题

request.setCharacterEncoding(“UTF-8”);

response.setContentType(“text/html;charset=UTF-8”);

1 源码解析

@Nullable
	private String encoding;

	private boolean forceRequestEncoding = false;

	private boolean forceResponseEncoding = false;

    public void setForceEncoding(boolean forceEncoding) {
       this.forceRequestEncoding = forceEncoding;
       this.forceResponseEncoding = forceEncoding;
    }

	@Override
	protected void doFilterInternal(
			HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
			throws ServletException, IOException {

		String encoding = getEncoding();
		if (encoding != null) {
			if (isForceRequestEncoding() || request.getCharacterEncoding() == null) {
				request.setCharacterEncoding(encoding);
			}
			if (isForceResponseEncoding()) {
				response.setCharacterEncoding(encoding);
			}
		}
		filterChain.doFilter(request, response);
	}

2 示例代码

<!--    注册CharacterEncodingFilter,解决请求与响应乱码-->
<filter>
    <filter-name>CharacterEncodingFilter</filter-name>
    <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <init-param>
        <param-name>forceEncoding</param-name>
        <param-value>true</param-value>
    </init-param>
    <init-param>
        <param-name>encoding</param-name>
        <param-value>UTF-8</param-value>
    </init-param>
</filter>
<filter-mapping>
    <filter-name>CharacterEncodingFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

** 注意:如有其它过滤器一同使用时,需要将CharacterEncodingFilter注册在第一位置**

SpringMVC中处理响应数据

JavaWeb响应数据基本流程

  1. 将数据共享到域中
    • pageContext
    • request:请求域
    • session:会话域
    • application【ServletContext】
  2. 跳转路径
    • 转发:request.getDispatcherServlet(url).forward(request,response);
    • 重定向:response.sendRedirect(url);

1 SpringMVC提供两种方式,处理响应数据

  • 使用ModelAndView作为方法返回值类型,处理响应数据
  • 使用ModelMapModelMap作为方法参数入参,处理响应数据

2 源码解析ModelAndView

public class ModelAndView {

   /** View instance or view name String. */
   @Nullable
   //视图对象
   private Object view;

   /** Model Map. */
   @Nullable
   //数据模型
   private ModelMap model;
   
   
   /**
   		设置&获取视图名称
   */
   public void setViewName(@Nullable String viewName) {
		this.view = viewName;
	}
	public String getViewName() {
		return (this.view instanceof String ? (String) this.view : null);
	}
	
    /**
    	获取数据模型
    */
    public Map<String, Object> getModel() {
		return getModelMap();
	}
    public ModelMap getModelMap() {
		if (this.model == null) {
			this.model = new ModelMap();
		}
		return this.model;
	}
    protected Map<String, Object> getModelInternal() {
		return this.model;
	}
    
    /**
    	设置模型数据
    */
    public ModelAndView addObject(Object attributeValue) {
		getModelMap().addAttribute(attributeValue);
		return this;
	}
    public ModelAndView addObject(String attributeName, @Nullable Object attributeValue) 	{
		getModelMap().addAttribute(attributeName, attributeValue);
		return this;
	}
   
}

3 使用ModelAndView作为方法返回值,处理响应数据【示例代码】

@RequestMapping("/responsedata")
public ModelAndView testResponseData(){
    ModelAndView mv = new ModelAndView();
    //为mv设置数据模型
    mv.addObject("deptId",1001);
    mv.addObject("deptName","秘书部门");
    //为mv设置视图对象
    mv.setViewName("success");
    return mv;
}

4使用Model或Map或ModelMap作为方法参数入参,处理响应数据【示例代码】

    @RequestMapping("/responsedataMap")
    public String testresponseDataMap(/*Map map*/Model model){

        //将数据共享域中【map】
        Emp emp = new Emp();
        emp.setEmpId(101);
        emp.setEmpName("康江");
        emp.setSalary(15000.0);
        emp.setDept(new Dept(1001,"研发部门"));

//        map.put("emp",emp);
        model.addAttribute("emp",emp);

        return "success";
    }

html获取响应数据

<h3>员工信息</h3>
<span th:text="${emp}"></span><br>
员工id:<span th:text="${emp.empId}"></span><br>
员工姓名:<span th:text="${emp.empName}"></span><br>
员工薪资:<span th:text="${emp.salary}"></span><br>
员工所属部门id:<span th:text="${emp.dept.id}"></span><br>
员工所属部门名称:<span th:text="${emp.dept.deptName}"></span><br>
  开发工具 最新文章
Postman接口测试之Mock快速入门
ASCII码空格替换查表_最全ASCII码对照表0-2
如何使用 ssh 建立 socks 代理
Typora配合PicGo阿里云图床配置
SoapUI、Jmeter、Postman三种接口测试工具的
github用相对路径显示图片_GitHub 中 readm
Windows编译g2o及其g2o viewer
解决jupyter notebook无法连接/ jupyter连接
Git恢复到之前版本
VScode常用快捷键
上一篇文章      下一篇文章      查看所有文章
加:2021-10-19 12:05:49  更:2021-10-19 12:07:14 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年12日历 -2024/12/23 18:01:08-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码
数据统计