服务网关 Gateway

服务网关 Gateway

Spring Cloud Gateway 是 Spring Cloud 生态系统中的网关,它基于 Spring5.0 + SpringBoot2.0 + WebFlux(基于高性能的 Reactor 模式响应式通信框架 Netty,异步非阻塞模型)等技术开发。旨在为微服务架构提供一种简单有效的、统一的 API 路由管理方式。其目标是代替 Zuul。

Spring Cloud Gateway 不仅提供统一的路由方式(反向代理)并且基于 Filter 链的方式提供了网关基本的功能,如:鉴权、流量控制、熔断、路径重写、日志监控等。

1. 核心概念

当用户发起一个请求之后,网关会根据一定的条件匹配,匹配成功之后可以将请求转发到指定的服务地址。而在这个过程中,我们可以进行一些具体的控制(限流、日志等)。

Gateway 有三个核心概念,分别是:路由、断言、过滤器。

  • 路由(Route)

网关的基本构建块,也是网关的基础工作单元。路由由一个 ID(唯一标识)、一个目标 RUL(最终路由到的地址)、一组断言(匹配条件判断)和一组过滤器(精细化控制)构成。如果断言为 true,则匹配该路由。

  • 断言(Predicate)

用于匹配 HTTP 请求的条件,它可以基于请求的路径、方法、参数、头部信息等多个因素进行判断(类似于 Nginx 中的 location 匹配一样),如果断言与请求相匹配则路由。

  • 过滤器(Filter)

用于对请求和响应进行处理的组件,它可以用于修改请求或响应的头部信息、添加请求或响应的字段、限流、重试等多种场景。总而言之,使用过滤器能够在请求之前或之后执行业务逻辑。

首先任何请求进来,网关都会把它们拦住。根据请求的URL把它们分配到不同的路由上,路由上面会有断言,来判断请求能不能进来。进来之后会有一系列的过滤器对请求被转发前或转发后进行改动。 具体怎么个改动法,那就根据业务不同而自定义了。一般就是监控,限流,日志输出等等。

2. 执行流程

Gateway 的执行流程如下:

在这里插入图片描述

  • 网关控制器映射(Gateway Handler Mapping)

找到与请求相匹配的路由,将其发送到网关 Web 控制器。

  • 网关 Web 控制器(Gateway Web Handler)

通过指定的过滤器将请求发送到我们实际的服务执行业务逻辑,然后返回。

  • 过滤器(Filter)

主要在请求之前(鉴权、参数校验、日志记录等)和之后(修改响应头、流量监控等)做一些控制处理

Spring Cloud Gateway 的执行流程简述:

客户端向 GateWay 发出请求,然后在 GateWay Handler Mapping 中找到与请求相匹配的路由,将其发送到 GateWay Web Handler;Handler 再通过指定的过滤器链来将请求发送到我们实际的服务执行业务逻辑,然后返回。

过滤器之间用虚线分开是因为过滤器可能会在发送代理请求之前(pre)或者之后(post)执行业务逻辑。

3. 入门案例

新建 gateway-9999 模块,然后在 pom.xml 文件中加入以下依赖:

<dependency>
    <groupId>org.springframework.cloud</groupId>
    <artifactId>spring-cloud-starter-gateway</artifactId>
</dependency>
<!-- 加入上面 Eureka 客户端的相关依赖,这里就不在展示~ -->

需注意:spring-boot-starter-web 依赖需要去除,否则会报错。

写 application.yaml 配置文件,具体如下:

server:
  port: 9999
eureka:
  client:
    service-url:
      defaultZone: http://localhost:7001/eureka/
  instance:
    instance-id: gateway-9999
spring:
  application:
    name: cloud-gateway-9999
  cloud:
    gateway:
      routes:
        # 路由ID,没有固定规则但要求唯一,建议配合服务名
        - id: provider-8001
          # 匹配后提供服务的路由地址
          uri: http://127.0.0.1:8001
          # 断言
          predicates:
            # 路径相匹配的进行路由
            - Path=/provider/**

在启动类上面加入 @EnableEurekaClient 注解,注册到 Eureka 注册中心,从注册中心获取对应服务。

配置好之后,我们启动注册中心 Eureka7001 和提供者 8001 服务,然后启动 Gateway 服务。

通过 Postman 访问 localhost:9999/provider/get/port 或者 localhost:9999/provider/index,经测试都可以访问。

还可以使用 Java API 的方式配置路由,创建 GatewayConfig 配置类,具体如下:

@Configuration
public class GatewayConfig {
    @Bean
    public RouteLocator getRouteLocator(RouteLocatorBuilder routeLocatorBuilder) {
        RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();
        /**
         * 设置路由
         *  id: 路由ID,要求唯一
         *  path(): gateway访问的路径
         *  uri(): 实际的服务地址
         */
        routes.route("my_route", item -> item.path("/guonei").uri("https://news.baidu.com")).build();
        routes.route("provider", item -> item.path("/provider/index").uri("http://127.0.0.1:8001")).build();
        return routes.build();
    }
}
4. 路由规则

一个请求在抵达网关层后,首先就要进行断言匹配,在满足所有断言之后才会进入 Filter 阶段。说白了 Predicate 就是一种路由规则,通过 Gateway 中丰富的内置断言的组合,我们就能让一个请求找到对应的 Route 来处理。

Spring Cloud GateWay 帮我们内置了很多 Predicates 功能,实现了各种路由匹配规则(通过Header、请求参数等作为条件)匹配到对应的路由。如下图所示:

在这里插入图片描述

  • DateTime 时间类断言

    利用时间断言可以实现一些类似于准点秒杀、商品限时特惠等场景。

    • After 时间点后匹配

    在该日期时间之后发生的请求都将被匹配。示例配置如下:

    spring:
      application:
        name: cloud-gateway-9999
      cloud:
        gateway:
          routes:
            - id: provider-8001
              uri: http://127.0.0.1:8001
              # 断言
              predicates:
                # 路径相匹配的进行路由
                # - Path=/provider/**
                - After=2024-01-01T12:00:32.077075700+08:00[Asia/Shanghai]
    

    配置文件中的时间参数为 UTC 时间格式,可以使用 System.out.println(ZonedDateTime.now()); 获取当前的 UTC 格式的时间。

    • Before 时间点前匹配

    在该日期时间之前发生的请求都将被匹配。示例配置如下:(下面配置代码只展示 predicates 部分~)

    predicates:
      - Before=2024-01-01T12:00:32.077075700+08:00[Asia/Shanghai]
    
    • Between 时间区间匹配

    有两个参数,datetime1 和 datetime2。在 datetime1 和 datetime2 之间的请求将被匹配。datetime2 参数的实际时间必须在 datetime1 之后。示例配置如下:

    predicates:
      - Between=2024-01-01T12:00:32.077075700+08:00[Asia/Shanghai],2024-01-02T12:00:32.077075700+08:00[Asia/Shanghai]
    
  • Cookie 类断言

验证 Cookie 中保存的信息,必须连同属性值一同验证,不能单独只验证属性是否存在。示例配置如下:

predicates:
  - Cookie=username,lyl

使用 Postman 进行接口测试,在 Headers 栏中加入 Key 为 Cookie,Value 为 username=lyl 的参数,然后访问对应接口。

  • Header 请求头类断言

Header 中包含了响应的属性才会被匹配,通常可以用来验证请求是否携带了访问令牌。示例配置如下:

predicates:
  # 请求头要有X-Request-Id属性并且值为整数的正则表达式
  - Header=X-Request-Id, \d+
  • Host 请求主机类断言

有一个参数:host name 列表。使用 Ant 路径匹配规则,点 . 作为分隔符。多个用逗号 , 隔开。示例配置如下:

predicates:
  - Host=lylhost

需要在本地 Host 文件中新增:127.0.0.1 lylhost

  • Method 请求方法类断言

HTTP 方法符合要求则匹配,例如 GET、POST 等,多个用逗号隔开,示例配置如下:

predicates:
  - Method=GET
  • Path 请求路径类断言

请求路径符合规则则匹配,支持正则表达式,示例配置如下:

predicates:
  - Path=/provider/**
  • QueryParam 请求参数类断言

验证参数,可单独验证是否包含参数,也可验证参数是否满足指定规则,符合则匹配,示例配置如下:

predicates:
  # 参数username符合要求则匹配
  - Query=username, \d+
  # 参数password存在则匹配
  - Query=password
  • RemoteAddr 远程地址类断言

远程地址符合则匹配,示例配置如下:

predicates:
  - RemoteAddr=192.168.1.1/24

动态路由

GateWay 支持自动从注册中心中获取服务列表并访问,即所谓的动态路由。具体配置如下:

spring:
  cloud:
    gateway:
      routes:
        - id: provider-8001
          # 匹配后提供服务的路由地址,lb后跟提供服务的微服务的名
          uri: lb://CLOUD-PROVIDER-8001
          predicates:
            - Path=/provider/**

动态路由设置时,uri 以 lb: // 开头(lb 代表从注册中心获取服务),后面是需要转发到的服务名称

分别启动 Eureka、provider-8001 和 provider-8002 服务,使用 Postman 调用 localhost:9999/provider/get/port 接口进行测试。

5. 网关过滤器

根据生命周期可以将 Spring Cloud Gateway 中的 Filter 分为 PRE 和 POST 两种:

  • PRE:代表在请求被路由之前执行该过滤器,可用来实现参数校验、权限校验、流量监控、日志输出、协议转换等功能。
  • POST:代表在请求被路由到微服务之后执行该过滤器。可用来实现响应头的修改(如添加标准的 HTTP Header )、收集统计信息和指标、将响应发送给客户端、输出日志、流量监控等功能。

根据作用范围,Filter 可以分为以下两种:

  • GatewayFilter:网关过滤器,只应用在单个路由或者一个分组的路由上。
  • GlobalFilter:全局过滤器,应用在所有的路由上。

GlobalFilter 全局过滤器是我们使用比较多的过滤器。

(1)GatewayFilter

网关过滤器(GatewayFilter)允许以某种方式修改传入的 HTTP 请求,或输出的 HTTP 响应。网关过滤器作用于特定路由。

Spring Cloud Gateway 内置了许多网关过滤器工厂来编写网关过滤器。例如:

  • AddRequestHeader:为原始请求添加 Header,参数为 Header 的名称及值
  • AddRequestParameter:为原始请求添加请求参数,参数为参数名称和值
  • AddResponseHeader:为原始响应添加 Header,参数为 Header 的名称及值
  • PrefixPath:为原始请求路径添加前缀 ,参数为前缀路径
  • RedirectTo:将原始请求重定向到指定的URL,参数为 HTTP 状态码及重定向的 URL
  • RemoveRequestHeader:为原始请求删除某个 Header,参数为 Header 名称
  • RemoveResponseHeader:为原始响应删除某个Header,参数为 Header 名称
  • SetResponseHeader:修改原始响应中某个 Header 的值,参数为 Header 名称和修改后的值
  • SetStatus:修改原始响应的状态码,参数为 HTTP 状态码,可以是数字,也可以是字符串

示例配置如下:

spring:
  cloud:
    gateway:
      routes:
        - id: provider-8001
          uri: lb://CLOUD-PROVIDER-8001
          predicates:
            - Path=/provider/**
          # 过滤器
          filters:
            # 修改原始响应的状态码
            - SetStatus=2024

我们也可以通过继承 AbstractGatewayFilterFactory 类,重写 shortcutFieldOrder() 方法的方式自定义网关过滤器,这里就不再展示~

(2)GlobalFilter

全局过滤器作用于所有路由,无需配置。通过全局过滤器可以实现对权限的统一校验,安全性验证等功能。

这里我们模拟一个黑名单拒绝访问的场景,具体示例代码如下:

/**
 * Description: 模拟实现黑名单功能
 * 场景:请求过来时,判断发送请求的客户端的ip,如果在黑名单中,拒绝访问
 */
@Slf4j
@Component
public class BlackListFilter implements GlobalFilter, Ordered {
    /**
     * 模拟黑名单(实际场景可从数据库或Redis中查询)
     */
    private static List<String> blackList = new ArrayList<>();

    static {
        // 模拟本机地址
        blackList.add("0:0:0:0:0:0:0:1");
        blackList.add("127.0.0.1");
    }

    /**
     * 过滤器核心方法
     * @param exchange 封装了request和response对象的上下文
     * @param chain    网关过滤器链(包含全局过滤器和单路由过滤器)
     */
    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        // 获取request和response对象
        ServerHttpRequest request = exchange.getRequest();
        ServerHttpResponse response = exchange.getResponse();
        // 获取客户端IP
        String clientIp = request.getRemoteAddress().getHostString();
        // 在黑名单中则拒绝访问,直接返回
        if (blackList.contains(clientIp)) {
            response.setStatusCode(HttpStatus.UNAUTHORIZED);
            log.info("=====>IP:" + clientIp + " 在黑名单中,将被拒绝访问!");
            String data = "Request be denied!";
            DataBuffer wrap = response.bufferFactory().wrap(data.getBytes());
            return response.writeWith(Mono.just(wrap));
        }
        // 不在黑名单的请求放行,执行后续的过滤器链
        return chain.filter(exchange);
    }

    /**
     * 返回值表示当前过滤器的顺序(优先级),数值越小,优先级越高
     */
    @Override
    public int getOrder() {
        return 0;
    }
}

通过 Postman 进行测试,返回 Request be denied!

6. 跨域配置

当一个请求 URL 的协议、域名、端口三者之间任意一个与当前页面 URL 不同即为跨域。

为什么会出现跨域:

出于浏览器的同源策略限制。同源策略是一种约定,它是浏览器最核心也最基本的安全功能,如果缺少了同源策略,则浏览器的正常功能可能都会受到影响。可以说 Web 是构建在同源策略基础之上的,浏览器只是针对同源策略的一种实现。

当前页面URL被请求页面URL是否跨域原因
http://www.test.com/http://www.test.com/index.html同源(协议、域名、端口号相同)
http://www.test.com/https://www.test.com/index.html协议不同(http/https)
http://www.test.com/http://www.baidu.com/主域名不同(test/baidu)
http://www.test.com/http://blog.test.com/子域名不同(www/blog)
http://www.test.com:8080/http://www.test.com:7001/端口号不同(8080/7001)

打开浏览器的 F12,当出现 ‘Access-Control-Allow-Origin’ 时,就是出现了跨域情况。

CORS

如何允许跨域,一种解决方法就是目的域告诉请求者允许什么来源域来请求,那么浏览器就会知道 B 域是否允许 A 域发起请求。

CORS(Cross-origin resource sharing),跨域资源共享就是这样一种解决手段。

CORS 使得浏览器在向目的域发起请求之前先发起一个 OPTIONS 方式的请求到目的域获取目的域的信息,比如获取目的域允许什么域来请求的信息。具体配置如下:

spring:
  cloud:
    gateway:
      globalcors:
        cors-configurations:
          '[/**]':
            allowCredentials: true
            allowedOriginPatterns: "*"
            allowedMethods: "*"
            allowedHeaders: "*"
        add-to-simple-url-handler-mapping: true
7. JWT

JWT(JSON Web Token)是一种用于双方之间传递安全信息的简洁的、URL 安全的声明规范。定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 对象的形式安全的传递信息。特别适用于分布式站点的单点登录(SSO)场景。

JWT 的原理就是,服务器在用户登录认证之后,生成一个 JSON 对象,然后生成一个长字符串返回给用户。用户在后面发起请求时,都会在请求头中携带 JTW,服务器对其进行验证,解析成功即让其成功访问,反之则拒绝访问。

JWT 是一个很长的字符串,中间用点.分割成 3 部分,示例如下:

eyJhbGciOiJIUzUxMiIsInR5cCI6IkpXVCJ9.eyJuYW1lIjoiSm9obiIsImFkbWluIjp0cnVlfQ.1YiYlxYX9I1Yjdk0lyJp3vluWfbOSsGToWzMDJcPcOpnbEVZZGzR4NhpgYFKqmAzarsfnNpOKrvS48KeXuO0IA

  • Header(头部)

JSON对象,描述 JWT 的元数据。

Header 通常包含两个部分,即 token 的类型(typ)和算法(alg)。

其中 alg 属性表示签名的算法(algorithm),默认是 HMAC SHA256(写成 HS256);typ 属性表示这个令牌(token)的类型(type),统一写为 JWT。示例如下:

{
  "alg": "HS512",
  "typ": "JWT"
}

Header 由 Base64 编码后构成了 JWT 的第一部分。

Base64 是一种用 64 个字符来表示任意二进制数据的方法,Base64 是一种任意二进制到文本字符串的编码方法,常用于在 URL、Cookie、网页中传输少量二进制数据。

  • Payload(载荷)

    Payload 是 JWT 的第二部分,它包含了要传递的信息,也就是所谓的 claim,有三种类型:

    • 预定义的声明(Registered Claims)

      这些声明是可选的,但建议尽可能使用。

      • iss:签发者
      • sub: 主题
      • aud:接收者
      • exp:过期时间,必须要大于签发时间
      • nbf:生效时间
      • iat:签发时间
      • jti:编号,唯一身份标识
    • 自定义的声明(Public Claims)

    自定义公共的声明,用于传递用户的信息或其他数据,建议使用命名空间来避免冲突。

    • 私有的声明(Private Claims)

    在双方约定的情况下私下使用的声明,避免和其他声明冲突。

    示例如下:

    {
        "sub":"GE_ecd48edee025487897926cdd1e71234",
        "name":"用户名",
        "created":1704092559000,
        "exp":1704178959
    }
    

    Payload 也是由 Base64 编码后与 Header 通过一个点号连接,形成了 JWT 的第二部分。

  • Signature(签证)

Signature 是 JWT 的第三部分,由 Header 和 Payload 经过 base64UrlEncode 之后,指定一个密钥(secret)通过算法计算得出,算法通常是 HS256(HMACSHA256)或RS256(RSASHA256)等。

HMACSHA512(base64UrlEncode(header) + "." + base64UrlEncode(payload), secret)

JWT 的签名过程如下:

1)取得 Header 和 Payload 的 Base64Url 编码后的字符串,中间用点号连接起来。

2)使用 Header 中指定的算法和密钥对上一步骤得到的字符串进行签名,生成 Signature。

3)将 Header、Payload 和 Signature 以点号连接起来,形成最终的 JWT。

在验证 JWT 时,需要对 Header 和 Payload 进行 Base64Url解码,再按照相同的算法和密钥计算Signature,与JWT中的Signature进行比对,如果一致说明JWT是真实的,否则说明JWT可能被篡改过。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/327819.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

如何在CentOS 7 中搭建Python 3.0 环境

1、下载 通过https://www.python.org/ftp/python/下载Python安装包&#xff0c;这里下载Python-3.10.9.tgz&#xff1b; 2、上传 借助MobaXterm等工具将Python安装包上传至/opt目录&#xff1b; 3、解压 将JDK压缩文件解压至/opt目录&#xff1a;tar -xvf /opt/Python-3.1…

React Store及store持久化的使用

1.安装 npm insatll react-redux npm install reduxjs/toolkit npm install redux-persist2. 使用React Toolkit创建counterStore并配置持久化 store/modules/counterStore.ts&#xff1a; import { createSlice } from reduxjs/toolkit// 定义状态类型 interface Action {…

Python数据分析案例33——新闻文本主题多分类(Transformer, 组合模型) 模型保存

案例背景 对于海量的新闻&#xff0c;我们可能需要进行文本的分类。模型构建很重要&#xff0c;现在对于自然语言处理基本都是神经网络的方法了。 本次这里正好有一组质量特别高的新闻数据&#xff0c;涉及 教育 科技 社会 时政 财经 房产 家居 七大主题&#xff0c;基本涵盖…

编译原理1.1习题 语言处理器

图源&#xff1a;文心一言 编译原理习题整理~&#x1f95d;&#x1f95d; 作为初学者的我&#xff0c;这些习题主要用于自我巩固。由于是自学&#xff0c;答案难免有误&#xff0c;非常欢迎各位小伙伴指正与讨论&#xff01;&#x1f44f;&#x1f4a1; 第1版&#xff1a;自…

HCIP -- ospf实验

要求&#xff1a; 实现&#xff1a; R3&#xff1a;int t 0/0/0ip address 172.16.1.4 255.255.255.248 &#xff08;配置虚拟接口ip地址&#xff09;tunnel-protocol gre p2mp &#xff08;配置接口协议为p2mp&#xff09;source 43.0.0.2 &#xff08;配置源&#xff09;osp…

【UE 材质】简单的纹理失真、溶解效果

目录 1. 失真效果 2. 溶解效果 3. 失真溶解 我们一开始有这样一个纹理 1. 失真效果 其中纹理节点“DistortTexture”的纹理为引擎自带的纹理“T_Noise01”&#xff0c;我们可以通过控制参数“失真度”来控制纹理的失真程度 2. 溶解效果 3. 失真溶解

奇安信天擎 rptsvr 任意文件上传漏洞复现

0x01 产品简介 奇安信天擎是奇安信集团旗下一款致力于一体化终端安全解决方案的终端安全管理系统(简称“天擎”)产品。通过“体系化防御、数字化运营”方法,帮助政企客户准确识别、保护和监管终端,并确保这些终端在任何时候都能可信、安全、合规地访问数据和业务。天擎基于…

微软推出付费版Copilot

关注卢松松&#xff0c;会经常给你分享一些我的经验和观点。 微软已经超越苹果&#xff0c;成了全球市值最高的公司&#xff0c;其他公司都因为AI大裁员&#xff0c;而微软正好相反&#xff0c;当然这个原因很简单&#xff1a;就是微软强制把AI全面接入到系统里来了。而Copilot…

Electron+React项目打包踩坑记录

首先&#xff0c;如何打包 写下本文的时间是 2024/01/16&#xff0c;搜索了网络上 ElectronReact 的打包方式&#xff0c;中间行不通&#xff0c;本文采用的方式是记录本文时 Electron 快速入门(https://www.electronjs.org/zh/docs/latest/tutorial/quick-start)记录的打包方式…

【Java 设计模式】创建型之抽象工厂模式

文章目录 1. 定义2. 应用场景3. 代码实现4. 应用示例结语 在软件开发中&#xff0c;抽象工厂模式是一种常见的创建型设计模式&#xff0c;它提供了一种创建一系列相关或相互依赖对象的接口&#xff0c;而无需指定它们具体的类。抽象工厂模式的核心思想是将一组相关的产品组合成…

spring boot shardingsphere mybatis-plus druid mysql 搭建mysql数据库读写分离架构

spring boot shardingsphere mybatis-plus druid mysql 搭建mysql数据库读写分离架构 ##关于window mysql主从搭建简单教程 传送门 window mysql5.7 搭建主从同步环境-CSDN博客 ##父pom.xml <?xml version"1.0" encoding"UTF-8"?> <project…

北交所交易手续费标准?哪家证券公司开通北交所券商交易手续费佣金万2?

北交所&#xff08;Beijing Exchange&#xff09;是指位于中国北京的一家金融交易所。北交所是中国政府为推动金融改革和国际化市场而设立的交易场所。它提供包括股票、债券、期货、外汇等多种金融产品的交易服务。北交所的目标是促进中国金融市场的发展&#xff0c;吸引国内外…

机器人持续学习基准LIBERO系列7——计算并可视化点云

0.前置 机器人持续学习基准LIBERO系列1——基本介绍与安装测试机器人持续学习基准LIBERO系列2——路径与基准基本信息机器人持续学习基准LIBERO系列3——相机画面可视化及单步移动更新机器人持续学习基准LIBERO系列4——robosuite最基本demo机器人持续学习基准LIBERO系列5——…

git项目管理

Git工作流程图 git 基础指令 git init #创建本地仓库,创建成功后&#xff0c;当前目录会多一个.git文件夹 git status #查看修改状态 git add . #添加工作区到暂存区 git commit -m 注释内容 #提交暂存区到本地仓库&#xff08;commit&#xff09; git log …

PngToIco.java

PngToIco.java [PNG转ICO]为了解决今天项目PNG弄成ICO写的 package image;import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException;import javax.imageio.ImageIO;import org.apache.commons.imaging.ImageFormats; import org.apache.common…

Git版本控制——分支

分支 几乎所有的版本控制系统都以某种形式支持分支。 使用分支意味着可以把工作从开发主线上分离开来进行重大的Bug修改、开发新的功能&#xff0c;以免影响开发主线。 查看本地分支 git branch创建本地分支 git branch 分支名切换分支(checkout) git checkout 分支名创建…

链表练习 Leetcode234.回文链表

题目传送门&#xff1a;Leetcode234 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为回文链表。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 1&#xff1a; 输入&#xff1a;head [1,2,2,1] 输出&#xff1a;true示例 2&…

x-cmd pkg | mermaid - 流程图、时序图等图表绘制工具

简介 mermaid-cli 是由 Mermaid 官方提供的命令行工具&#xff0c;用于将 Mermaid 语法的文本转换为 SVG / PNG / PDF。 Mermaid 是一个基于 JavaScript 的图表绘制工具&#xff0c;它使用简单的文本描述语法&#xff0c;就可以绘制出流程图、时序图、甘特图等多种图表。 首次…

FFmpeg连载6-音频重采样

今天我们的实战内容是将音频解码成PCM&#xff0c;并将PCM重采样成特定的采样率&#xff0c;然后输出到本地文件进行播放。 什么是重采样&#xff1f; 所谓重采样&#xff0c;一句话总结就是改变音频的三元素&#xff0c;也就是通过重采样改变音频的采样率、采样格式或者声道数…

一文了解ChatGPT4+Python近红外光谱数据分析及机器学习与深度学习建模应用

2022年11月30日&#xff0c;可能将成为一个改变人类历史的日子——美国人工智能开发机构OpenAI推出了聊天机器人ChatGPT3.5&#xff0c;将人工智能的发展推向了一个新的高度。2023年4月&#xff0c;更强版本的ChatGPT4.0上线&#xff0c;文本、语音、图像等多模态交互方式使其在…