| REST:Representational State Transfer,表现层资源状态转移。 |
| a>资源 |
| 资源是一种看待服务器的方式,即,将服务器看作是由很多离散的资源组成。每个资源是服务器上一个 |
| 可命名的抽象概念。因为资源是一个抽象的概念,所以它不仅仅能代表服务器文件系统中的一个文件、 |
| 数据库中的一张表等等具体的东西,可以将资源设计的要多抽象有多抽象,只要想象力允许而且客户端 |
| 应用开发者能够理解。与面向对象设计类似,资源是以名词为核心来组织的,首先关注的是名词。一个 |
| 资源可以由一个或多个URI来标识。URI既是资源的名称,也是资源在Web上的地址。对某个资源感兴 |
| 趣的客户端应用,可以通过资源的URI与其进行交互。 |
| b>资源的表述 |
| 资源的表述是一段对于资源在某个特定时刻的状态的描述。可以在客户端-服务器端之间转移(交 |
| 换)。资源的表述可以有多种格式,例如HTML/XML/JSON/纯文本/图片/视频/音频等等。资源的表述格 |
| 式可以通过协商机制来确定。请求-响应方向的表述通常使用不同的格式。 |
| c>状态转移 |
| 状态转移说的是:在客户端和服务器端之间转移(transfer)代表资源状态的表述。通过转移和操作资 |
| 源的表述,来间接实现操作资源的目的 |
| |
| 四种基本操作:GET 用来获取资源,POST 用来新建资源,PUT 用来更新资源,DELETE |
| 用来删除资源 |
| REST 风格提倡 URL 地址使用统一的风格设计,从前到后各个单词使用斜杠分开,不使用问号键值对方 |
| 式携带请求参数,而是将要发送给服务器的数据作为 URL 地址的一部分,以保证整体风格的一致性 |
| # springmvc配置文件中配置如下,用于跳转到test_rest.html |
| <mvc:view-controller path="/test_rest" view-name="test_rest"></mvc:view-controller> |
| |
| # SpringMVC 提供了 HiddenHttpMethodFilter 帮助我们将 POST 请求转换为 DELETE 或 PUT 请求 |
| # web.xml中配置如下 |
| <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> |
| |
| # 编写test_rest.html |
| <!DOCTYPE html> |
| <html lang="en" xmlns:th="http://www.thymeleaf.org"> |
| <head> |
| <meta charset="UTF-8"> |
| <title>Title</title> |
| </head> |
| <body> |
| <a th:href="@{/user}">查询所有用户信息</a><br> |
| <a th:href="@{/user/1}">根据id查询用户信息</a><br> |
| <form th:action="@{/user}" method="post"> |
| 用户名:<input type="text" name="username"><br> |
| 密码:<input type="password" name="password"><br> |
| <input type="submit" value="添加"><br> |
| </form> |
| <form th:action="@{/user}" method="post"> |
| <input type="hidden" name="_method" value="PUT"> |
| 用户名:<input type="text" name="username"><br> |
| 密码:<input type="password" name="password"><br> |
| <input type="submit" value="修改"><br> |
| </form> |
| </body> |
| </html> |
| |
| # 编写控制器 |
| @Controller |
| public class UserController { |
| |
| /** |
| * 使用RESTFul模拟用户资源的增删改查 |
| * /user GET 查询所有用户信息 |
| * /user/1 GET 根据用户id查询用户信息 |
| * /user POST 添加用户信息 |
| * /user/1 DELETE 删除用户信息 |
| * /user PUT 修改用户信息 |
| */ |
| @RequestMapping(value = "/user", method = RequestMethod.GET) |
| public String getAllUser(){ |
| System.out.println("查询所有用户信息"); |
| return "success"; |
| } |
| |
| @RequestMapping(value = "/user/{id}", method = RequestMethod.GET) |
| public String getUserById(){ |
| System.out.println("根据id查询用户信息"); |
| return "success"; |
| } |
| |
| @RequestMapping(value = "/user", method = RequestMethod.POST) |
| public String insertUser(String username, String password){ |
| System.out.println("添加用户信息:"+username+","+password); |
| return "success"; |
| } |
| |
| @RequestMapping(value = "/user", method = RequestMethod.PUT) |
| public String updateUser(String username, String password){ |
| System.out.println("修改用户信息:"+username+","+password); |
| return "success"; |
| } |
| |
| } |

- 查看HiddenHttpMethodFilter源码
| protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException { |
| HttpServletRequest requestToUse = request; |
| |
| if ("POST".equals(request.getMethod()) && request.getAttribute("javax.servlet.error.exception") == null) { |
| |
| String paramValue = request.getParameter(this.methodParam); |
| |
| if (StringUtils.hasLength(paramValue)) { |
| |
| String method = paramValue.toUpperCase(Locale.ENGLISH); |
| if (ALLOWED_METHODS.contains(method)) { |
| requestToUse = new HiddenHttpMethodFilter.HttpMethodRequestWrapper(request, method); |
| } |
| } |
| } |
| |
| filterChain.doFilter((ServletRequest)requestToUse, response); |
| } |
| |
| # 代码块详情 |
| if (ALLOWED_METHODS.contains(method)) { |
| requestToUse = new HiddenHttpMethodFilter.HttpMethodRequestWrapper(request, method); |
| } |
| # 以上代码表示获取到参数method中的值是否为如下几种请求方式 |
| private static final List<String> ALLOWED_METHODS; |
| static { |
| ALLOWED_METHODS = Collections.unmodifiableList(Arrays.asList(HttpMethod.PUT.name(), HttpMethod.DELETE.name(), HttpMethod.PATCH.name())); |
| } |
| |
| # 之后为我们创建一个新的对象,并将这个对象放行 |
| requestToUse = new HiddenHttpMethodFilter.HttpMethodRequestWrapper(request, method); |
| filterChain.doFilter((ServletRequest)requestToUse, response); |
| |
| # 创建新的对象的方法如下,即是将我们传入的请求方式作为新的请求方式 |
| private static class HttpMethodRequestWrapper extends HttpServletRequestWrapper { |
| private final String method; |
| |
| public HttpMethodRequestWrapper(HttpServletRequest request, String method) { |
| super(request); |
| this.method = method; |
| } |
| |
| public String getMethod() { |
| return this.method; |
| } |
| } |
| HiddenHttpMethodFilter 处理put和delete请求的条件: |
| a>当前请求的请求方式必须为post |
| b>当前请求必须传输请求参数_method |
| 满足以上条件,HiddenHttpMethodFilter 过滤器就会将当前请求的请求方式转换为请求参数 |
| _method的值,因此请求参数_method的值才是最终的请求方式 |
| # 过滤器配置是从上往下执行,所以配置字符集的过滤器应该卸载上面 |
| <filter> |
| <filter-name>CharacterEncodingFilter</filter-name> |
| <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class> |
| <init-param> |
| <param-name>encoding</param-name> |
| <param-value>UTF-8</param-value> |
| </init-param> |
| <init-param> |
| <param-name>forceResponseEncoding</param-name> |
| <param-value>true</param-value> |
| </init-param> |
| </filter> |
| <filter-mapping> |
| <filter-name>CharacterEncodingFilter</filter-name> |
| <url-pattern>/*</url-pattern> |
| </filter-mapping> |
| |
| <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> |
| # 当工程中的web.xml与tomcat中web.xml冲突时,以工程中的web.xml为准 |
| |
| # 查看tomcat中的web.xml,默认servlet在tomcat容器启动时初始化 |
| <servlet> |
| <servlet-name>default</servlet-name> |
| <servlet-class>org.apache.catalina.servlets.DefaultServlet</servlet-class> |
| <init-param> |
| <param-name>debug</param-name> |
| <param-value>0</param-value> |
| </init-param> |
| <init-param> |
| <param-name>listings</param-name> |
| <param-value>false</param-value> |
| </init-param> |
| <load-on-startup>1</load-on-startup> |
| </servlet> |
| |
| # 处理jsp访问的 |
| <servlet> |
| <servlet-name>jsp</servlet-name> |
| <servlet-class>org.apache.jasper.servlet.JspServlet</servlet-class> |
| <init-param> |
| <param-name>fork</param-name> |
| <param-value>false</param-value> |
| </init-param> |
| <init-param> |
| <param-name>xpoweredBy</param-name> |
| <param-value>false</param-value> |
| </init-param> |
| <load-on-startup>3</load-on-startup> |
| </servlet> |
| |
| # 默认servlet的映射信息 |
| <servlet-mapping> |
| <servlet-name>default</servlet-name> |
| <url-pattern>/</url-pattern> |
| </servlet-mapping> |
| |
| |
| <servlet-mapping> |
| <servlet-name>jsp</servlet-name> |
| <url-pattern>*.jsp</url-pattern> |
| <url-pattern>*.jspx</url-pattern> |
| </servlet-mapping> |
| |
| # 如下我们在自己工程中注册的servlet,与tomcat映射的路径冲突了,所以以自己工程的配置为准 |
| <servlet> |
| <servlet-name>DispatcherServlet</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> |
| </init-param> |
| <load-on-startup>1</load-on-startup> |
| </servlet> |
| <servlet-mapping> |
| <servlet-name>DispatcherServlet</servlet-name> |
| <url-pattern>/</url-pattern> |
| </servlet-mapping> |
| |
| # 所以我们在springmvc配置文件中配置如下,表示浏览器发送请求,我们的DispatcherServlet无法处理时,则交给tomcat默认的servlet处理 |
| # 如下2个标签需一起使用,如果只使用上面的标签,则所有的请求都会交给tomcat默认的servlet处理 |
| |
| <mvc:default-servlet-handler /> |
| |
| <mvc:annotation-driven /> |
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?