restful

本文详细探讨了RESTful API的最佳实践与应用场景,包括如何在不同场景下正确使用HTTP方法,如POST、PUT和DELETE,以及如何处理JSON格式的数据传输。特别关注了在Spring框架下,如何配置过滤器来解决PUT和DELETE请求的参数接收问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1. Rest最佳场景

Rest最佳实践,Json作为通信载体
在这里插入图片描述
在这里插入图片描述

2. Rest应用场景

如下3个场景中,主要讨论,在请求 Put Delete 时:如何正确请求 如何正确传参

情景1:发送Post请求,中途转为Put Delete
浏览器本身只支持 get 和 post 请求,作为客户端,使用Rest时,如果需要发送 put 和 delete 请求,则:
在这里插入图片描述
<!-- 1. 对于tomcat请求依然是post,对于Handler是Put -->
<form action="${pageContext.request.contextPath}/rest04/users" method="post">
    <input type="hidden" name="_method" value="put">
    <input type="hidden" name="id" value="1">
    <input type="text" name="name" value="zhj">
    <input type="text" name="gender" value="true">
    <input type="text" name="birth" value="2019-12-12">
    <input type="submit" value="修改">
</form>
<!-- 2. 设置此过滤器,作用:
        如果当前请求是Post,则获取请求中的"_method"参数值,并将请求的method转换为对应值。
 		即转换了请求的method
-->
<filter>
    <filter-name>httpMethodFilter</filter-name>
    <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>httpMethodFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
// 3. 定义domain
public class User {
    private Integer id;
    private String name;
    private Boolean gender;
    @DateTimeFormat(pattern = "yyyy-MM-dd")
    private Date birth;
}
// 4. 定义Handler,正常接收参数!!!!!
@RequestMapping(value="/users",method = RequestMethod.PUT)
@ResponseBody
//正常接收即可,因为此时发送的依然是post请求,tomcat已经获取到参数
public MyRequestStatus updateUser(User user) {
    System.out.println("update One user2:"+user);
    MyRequestStatus status = new MyRequestStatus("update", "ok");
    return status;
}

情景2:直接发送Put Delete请求

tomcat不处理put请求的请求体,其中参数不接收

var xhr = new XMLHttpRequest(); 
xhr.open("put","${pageContext.request.contextPath}/rest04/users");//ajax发送正常的put请求
xhr.setRequestHeader("content-type","application/x-www-form-urlencoded");
xhr.send("id=1&name=zhj&gender=true&birth=2019-12-12");//携带参数
// 2. 定义domain
public class User {
    private Integer id;
    private String name;
    private Boolean gender;
    @DateTimeFormat(pattern = "yyyy-MM-dd")
    private Date birth;
}
// 3.定义Handler,此时无法接收到请求参数!!!!!
@RequestMapping(value="/users",method = RequestMethod.PUT)
@ResponseBody
public MyRequestStatus updateUser(User user) {//此时无法接收到请求参数!!!!
    System.out.println("update One user2:"+user);
    MyRequestStatus status = new MyRequestStatus("update", "ok");
    return status;
}
<!-- 解决方案:
     如果Put请求中的 ”id=1&name=zhj&...“ 含有如此参数,则可以配置如下过滤器
     其作用:如果发现当前请求是Put请求,会将请求体中的请求参数获取出来,供后续使用。
-->
<filter>
    <filter-name>put</filter-name>
    <filter-class>org.springframework.web.filter.HttpPutFormContentFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>put</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

情景3:直接发送Put请求,但携带Json格式数据参数

var xhr = new XMLHttpRequest(); 
xhr.open("put","${pageContext.request.contextPath}/rest04/users");//ajax发送正常的put请求
xhr.setRequestHeader("content-type","application/json");
xhr.send('{"id":1,"name":"zhj","gender":true,"birth":"2019/12/12"}');//携带Json数据参数
public class User {
    private Integer id;
    private String name;
    private Boolean gender;
    @JsonFormat(pattern = "yyyy/MM/dd")
    private Date birth;
}
@RequestMapping(value="/users",method = RequestMethod.PUT)
@ResponseBody
public MyRequestStatus updateUser(@RequestBody User user) {//通过@RequestBody收参!!!!
    System.out.println("update One user2:"+user);
    MyRequestStatus status = new MyRequestStatus("update", "ok");
    return status;
}

3. RestComponent

RestFul Controller

资源:用户  /users   GET  POST  PUT
     某一个用户   /users/{id}  GET  DELETE
     分页用户  /users/{pageNum}/{pageSize}  GET

@Controller
public class RestController {
    @GetMapping("/users")
    @ResponseBody
    public List<User> queryAllUsers(){//查询所有用户
        System.out.println("get");
        List<User> users = ....
        return users;
    }

    @PostMapping("/users")
    public String addUser(User user){//增加用户
        System.out.println("Post user :"+user);
        return "index";
    }
	
	//更新用户
    @PutMapping("/users")
    public String updateUser(@RequestBody User user){
        System.out.println("Put user" user:"+user);
        return "index";
    }
    
    // 查询id为x的用户
    @GetMapping("/users/{id}")
    public String queryOneUser(@PathVariable Integer id){//@PathVariable 接收路径中的值
        System.out.println("Get user id:"+id);
        return "index";
    }
    
    //删除用户
    @DeleteMapping("/users/{id}")
    public String deleteOneUser(@PathVariable Integer id){//@PathVariable 接收路径中的值
        System.out.println("delete user id:"+id);
        return "index";
    }

    // 查询分页用户
    @GetMapping("/users/{pageNum}/{pageSize}")
    @ResponseBody
    public List<User> queryAllUsers(@PathVariable Integer pageNum,@PathVariable Integer PageSize){
        System.out.println("get");
        List<User> users = ....
        return users;
    }
}
03-19
### IEEE 802.1Q VLAN Tagging Protocol Standard IEEE 802.1Q 是支持虚拟局域网(VLAN)的标准协议之一,通常被称为 Dot1q。该标准定义了一种用于以太网帧的 VLAN 标记系统以及交换机和桥接器处理这些标记帧的操作流程[^2]。 #### 协议结构概述 IEEE 802.1Q 的核心功能在于通过在以太网数据帧中插入特定字段来实现 VLAN 标签的功能。这种标签使得网络设备能够识别哪些流量属于哪个 VLAN,并据此执行转发决策。具体来说: - **Tag Header**: 在原始以太网帧头部增加了一个额外的 4 字节字段作为 VLAN 标签头。这四个字节包含了以下部分: - **Priority Code Point (PCP)**: 使用 3 比特表示优先级级别,范围从 0 到 7,主要用于 QoS 控制。 - **Canonical Format Indicator (CFI)**: 这是一个单比特位,在传统以太网环境中设置为零。 - **VLAN Identifier (VID)**: 使用 12 比特标识具体的 VLAN ID,理论上可以支持多达 4096 个不同的 VLAN(编号从 0 至 4095),其中某些特殊值保留给内部用途或管理目的。 #### 数据包处理机制 当一个带有 VLAN tag 的数据包进入支持 IEEE 802.1Q 的交换机时,它会依据此标签决定如何路由或者过滤该数据流。如果目标端口不属于同一 VLAN,则不会传输至其他无关联的物理接口上;反之亦然——只有相同 VLAN 成员之间才允许互相通信除非经过路由器跨网段访问[^1]。 此外,为了简化管理和配置过程并增强互操作性,还引入了一些辅助性的子协议和服务组件比如 GARP(通用属性注册协议)。GARP 可帮助分发有关 VLAN 成员资格的信息到各个连接节点以便动态调整其行为模式而无需频繁手动干预[^3]。 以下是创建带 VLAN TAG 的 Python 示例代码片段展示如何模拟构建这样的 Ethernet Frame: ```python from scapy.all import Ether, Dot1Q, IP, sendp def create_vlan_packet(src_mac="00:aa:bb:cc:dd:ee", dst_mac="ff:ff:ff:ff:ff:ff", vlan_id=100, src_ip="192.168.1.1", dst_ip="192.168.1.2"): ether = Ether(src=src_mac, dst=dst_mac) dot1q = Dot1Q(vlan=vlan_id) ip_layer = IP(src=src_ip, dst=dst_ip) packet = ether / dot1q / ip_layer return packet packet = create_vlan_packet() sendp(packet, iface="eth0") # Replace 'eth0' with your network interface name. ``` 上述脚本利用 Scapy 库生成包含指定源地址、目的地址及所属 VLAN 编号的数据报文并通过选定的网卡发送出去测试实际效果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值