【微服务】Gateway

文章目录

    • 1.基本介绍
        • 官方文档:https://springdoc.cn/spring-cloud-gateway/#gateway-starter
        • 1.引出网关
        • 2.使用网关服务架构图
        • 3.Gateway网络拓扑图(背下来)
        • 4.Gateway特性
        • 5.Gateway核心组件
          • 1.基本介绍
          • 2.断言
          • 3.过滤
        • 6.Gateway工作机制
    • 2.搭建Gateway微服务
        • 1.引入Gateway架构
        • 2.环境搭建
          • 1.参考member-service-consumer-81 创建 e-commerce-gateway-20000 子模块
          • 2.检查父子的pom.xml
            • 1.父pom.xml
            • 2.子pom.xml
          • 3.pom.xml引入依赖
          • 4.application.yml 配置gateway(这里的eureka还没有用到只是先配置一下)
          • 5.编写启动类
          • 6.依次启动9001,20000,10001模块进行测试
            • 1.9001报错连接不上9002
            • 2.20000Gateway正常启动
            • 3.10001报错,连接9002被拒绝
            • 4.输入http://eureka9001.com:9001/查看注册中心状态,成功注册20000, 10001!
            • 5.postman测试
          • 7.修改application.yml增加Gateway路由,访问到/member/save
          • 8.postman测试
    • 3.配置类注入Gateway(了解)
        • 1.com/sun/springcloud/config/GatewayRoutesConfig.java
        • 2.postman测试
    • 4.动态路由
        • 1.需求分析
        • 2.动态路由配置
        • 3.启动服务
        • 4.postman测试
          • 1.第一次
          • 2.第二次,可以看出负载均衡算法默认是轮询
    • 5.配置负载均衡算法
        • 1.com/sun/springcloud/config/RibbonRule.java 配置类直接注入负载均衡算法的对象即可
        • 2.重启服务进行测试
          • 1.查看注册中心
          • 2.postman向gateway发送几次请求
            • 1.第一次
            • 2.第二次,可以发现两次一样,不再是轮询算法
            • 3.恢复轮询算法,注销掉刚才的配置类
    • 6.Gateway 细节说明
        • 1.关于predicates: - Path=/member/save中的`/`
        • 2.关于uri: lb://member-service-provider服务发现的内容
        • 3.Gateway执行流程
          • 1.断言
          • 2.服务发现
          • 3.路径拼接
          • 4.远程调用
    • 6.Predicate(断言)
        • 1.基本介绍
        • 2.After Route Predicate 某个时间之后发送请求
          • 1.需求分析
          • 2.代码获取Gateway格式的时间
          • 3.Gateway格式的时间
          • 4.或者直接从官网获取这个Gateway格式的时间
          • 5.应用实例
          • 6.启动服务发送请求
        • 3.Before Route Predicate 某个时间之前发送请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman发送请求
        • 4.Between Route Predicate 两个时间之间发送请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman发送请求
        • 5.Cookie Route Predicate 携带某个Cookie的请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman发送请求并设置cookie
            • 1.携带cookie
            • 2.不携带cookie
        • 6.Header Route Predicate 携带某个Header属性的请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman请求Header携带参数
        • 7.Host Route Predicate 指定主机发送的请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman请求修改host
        • 8.Method Route Predicate 指定请求方式的请求
          • 1.应用实例
          • 2.postman测试
        • 9.Path Route Predicate 指定路径的请求
          • 应用实例
        • 10.Query Route Predicate 参数满足指定条件的请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman测试
        • 11.RemoteAddr Route Predicate 指定范围的ip客户端发送的请求
          • 1.需求分析
          • 2.应用实例
          • 3.postman测试
        • 12.weight Route Predicate 根据权重进行断言
    • 7.过滤器
        • 1.基本介绍
          • 1.功能
          • 2.工作流程
        • 2.自定义过滤器
          • 1.需求分析
          • 2.代码实现 com/sun/springcloud/filter/CustomerGateWayFilter.java
          • 3.postman测试

1.基本介绍

官方文档:https://springdoc.cn/spring-cloud-gateway/#gateway-starter
1.引出网关

image-20240326082643333

2.使用网关服务架构图

image-20240326083139497

3.Gateway网络拓扑图(背下来)

image-20240326092944937

4.Gateway特性

image-20240326085401897

5.Gateway核心组件
1.基本介绍

image-20240326085927104

2.断言

3.过滤

image-20240326090047153

6.Gateway工作机制

image-20240326090617933

2.搭建Gateway微服务

1.引入Gateway架构

image-20240326091245301

2.环境搭建
1.参考member-service-consumer-81 创建 e-commerce-gateway-20000 子模块
2.检查父子的pom.xml
1.父pom.xml

image-20240326092211301

2.子pom.xml

image-20240326092252177

3.pom.xml引入依赖
    <dependencies><!--gateway不需要web-starter!--><!--&lt;!&ndash; springboot web starter 用来监听端口&ndash;&gt;--><!--<dependency>--><!--    <groupId>org.springframework.boot</groupId>--><!--    <artifactId>spring-boot-starter-web</artifactId>--><!--    &lt;!&ndash; 如果在子工程/模块指定了 version,则以指定为准 &ndash;&gt;--><!--</dependency>--><!--<dependency>--><!--    <groupId>org.springframework.boot</groupId>--><!--    <artifactId>spring-boot-starter-actuator</artifactId>--><!--</dependency>--><!-- 引入 cloud gateway --><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-gateway</artifactId></dependency><!-- 引入eureka的客户端场景启动器 --><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-netflix-eureka-client</artifactId><!-- 使用版本仲裁 --></dependency><!-- lombok --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><optional>true</optional></dependency><!-- test --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><!-- 公共模块的jar包 --><dependency><groupId>org.example</groupId><artifactId>e_commerce_center-common-api</artifactId><version>1.0-SNAPSHOT</version></dependency></dependencies>
4.application.yml 配置gateway(这里的eureka还没有用到只是先配置一下)
server:port: 20000
spring:application:name: e-commerce-gatewaycloud:gateway:routes: # 配置路由, List<RouteDefinition> routes,是集合类型- id: member_route01 # 路由id,唯一即可uri: http://localhost:10001 # 断言匹配成功后将这个与/member/get/**拼接predicates:- Path=/member/get/** # 断言匹配成功后的完整路径是http://localhost:10001/member/get/**
# 配置eureka,作为客户端,有服务注册和服务发现功能
eureka:instance:hostname: e-commerce-serviceclient:register-with-eureka: true # 将自己注册到eureka服务fetch-registry: true # 发现服务功能,如果是集群,必须要能发现服务才能配合ribben进行负载均衡service-url:defaultZone: http://eureka9001.com:9001/eureka/
5.编写启动类
package com.sun.springcloud;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;/*** @author 孙显圣* @version 1.0*/
@SpringBootApplication
@EnableEurekaClient
public class GateWayApplication20000 {public static void main(String[] args) {SpringApplication.run(GateWayApplication20000.class, args);}
}
6.依次启动9001,20000,10001模块进行测试
1.9001报错连接不上9002
  • 9002是eureka的一个server,与这个9001互相注册
  • 而9002并没有开启,所以连接不上是正常的

image-20240326104857595

2.20000Gateway正常启动

image-20240326105206507

3.10001报错,连接9002被拒绝
  • 10001是微服务集群中的一个服务,它注册到了9001和9002的server
  • 9002没开,所以连接被拒绝,没问题

image-20240326105355589

4.输入http://eureka9001.com:9001/查看注册中心状态,成功注册20000, 10001!

image-20240326105615850

5.postman测试

路径匹配的情况

image-20240326110159956

路径不匹配的情况,Gateway直接返回404

image-20240326110253187

7.修改application.yml增加Gateway路由,访问到/member/save
server:port: 20000
spring:application:name: e-commerce-gatewaycloud:gateway:routes: # 配置路由, List<RouteDefinition> routes,是集合类型- id: member_route01 # 路由id,唯一即可uri: http://localhost:10001 # 断言匹配成功后将这个与/member/get/**拼接predicates:- Path=/member/get/** # 断言匹配成功后的完整路径是http://localhost:10001/member/get/**- id: member_route02 # 路由id,唯一即可uri: http://localhost:10001 # 断言匹配成功后将这个与/member/save拼接predicates:- Path=/member/save # 断言匹配成功后的完整路径是http://localhost:10001//member/save# 配置eureka,作为客户端,有服务注册和服务发现功能
eureka:instance:hostname: e-commerce-serviceclient:register-with-eureka: true # 将自己注册到eureka服务fetch-registry: true # 发现服务功能,如果是集群,必须要能发现服务才能配合ribben进行负载均衡service-url:defaultZone: http://eureka9001.com:9001/eureka/
8.postman测试

image-20240326112118668

3.配置类注入Gateway(了解)

1.com/sun/springcloud/config/GatewayRoutesConfig.java
package com.sun.springcloud.config;import org.springframework.cloud.gateway.route.RouteLocator;
import org.springframework.cloud.gateway.route.builder.RouteLocatorBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;/*** Description:** @Author sun* @Create 2024/3/26 11:36* @Version 1.0*/
@Configuration
public class GatewayRoutesConfig {@Beanpublic RouteLocator routeLocator(RouteLocatorBuilder routeLocatorBuilder) {RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();// 这里使用了Lambda表达式,理解为规定写法即可return routes.route("member_route04",r -> r.path("/member/get/**").uri("http://localhost:10001")).build();}
}
2.postman测试

image-20240326123239832

4.动态路由

1.需求分析

image-20240326124342744

2.动态路由配置
server:port: 20000
spring:application:name: e-commerce-gatewaycloud:gateway:routes: # 配置路由, List<RouteDefinition> routes,是集合类型- id: member_route01 # 路由id,唯一即可uri: lb://member-service-provider # 断言匹配成功后将这个与/member/get/**拼接,这里的lb是负载均衡的协议,服务名需要小写predicates:- Path=/member/get/** # 断言匹配成功后的完整路径是http://localhost:10001/member/get/**- id: member_route02 # 路由id,唯一即可uri: lb://member-service-provider # 断言匹配成功后将这个与/member/save拼接,,这里的lb是负载均衡的协议,,服务名需要小写predicates:- Path=/member/save # 断言匹配成功后的完整路径是http://localhost:10001//member/savediscovery:locator:enabled: true # 启用服务发现,不配置可能也不影响# 配置eureka,作为客户端,有服务注册和服务发现功能
eureka:
#  instance:
#    hostname: e-commerce-serviceclient:register-with-eureka: true # 将自己注册到eureka服务fetch-registry: true # 发现服务功能,如果是集群,必须要能发现服务才能配合ribben进行负载均衡service-url:defaultZone: http://eureka9001.com:9001/eureka/
3.启动服务

image-20240326125734725

4.postman测试
1.第一次

image-20240326125806694

2.第二次,可以看出负载均衡算法默认是轮询

image-20240326125819039

5.配置负载均衡算法

1.com/sun/springcloud/config/RibbonRule.java 配置类直接注入负载均衡算法的对象即可
package com.sun.springcloud.config;import com.netflix.loadbalancer.IRule;
import com.netflix.loadbalancer.RandomRule;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;/*** Description: 配置自己的负载均衡算法,这里使用的随机算法** @Author sun* @Create 2024/3/26 14:31* @Version 1.0*/
@Configuration
public class RibbonRule {@Beanpublic IRule myRibbonRule() {return new RandomRule();}
}
2.重启服务进行测试
1.查看注册中心

image-20240326143837463

2.postman向gateway发送几次请求
1.第一次

image-20240326143910997

2.第二次,可以发现两次一样,不再是轮询算法

image-20240326143922073

3.恢复轮询算法,注销掉刚才的配置类

image-20240326144200220

6.Gateway 细节说明

1.关于predicates: - Path=/member/save中的/

这里的/指的是http://ip + 端口/上下文路径/

2.关于uri: lb://member-service-provider服务发现的内容

这里通过注册中心发现的是微服务集群中对应服务的http://ip+端口/上下文路径(如果配置了的话)

3.Gateway执行流程
1.断言

对请求url进行断言

2.服务发现

如果通过断言,则从注册中心进行服务发现,并通过负载均衡算法得到指定服务的http://ip+端口/上下文路径(如果配置了的话)

3.路径拼接

将得到的http://ip+端口/上下文路径(如果配置了的话)与进行断言的路径拼接

4.远程调用

通过拼接后的路径完成远程调用

6.Predicate(断言)

1.基本介绍

image-20240326150551698

2.After Route Predicate 某个时间之后发送请求
1.需求分析

image-20240326150812006

2.代码获取Gateway格式的时间
package com.sun.springcloud;
import java.time.ZonedDateTime;/*** Description:** @Author sun* @Create 2024/3/26 15:10* @Version 1.0*/
public class T2 {public static void main(String[] args) {ZonedDateTime now = ZonedDateTime.now();System.out.println(now);}
}
3.Gateway格式的时间
2024-03-26T15:12:23.371+08:00[Asia/Shanghai]

image-20240326151335684

4.或者直接从官网获取这个Gateway格式的时间
5.应用实例
          predicates:- Path=/member/get/**- After=2025-01-20T17:42:47.789+08:00[Asia/Shanghai] # 断言条件:在2025年1月20日下午5点42之后发送的请求
6.启动服务发送请求

image-20240326153327603

3.Before Route Predicate 某个时间之前发送请求
1.需求分析

image-20240326153651900

2.应用实例
          predicates:- Path=/member/get/** - Before=2025-01-20T17:42:47.789-07:00[America/Denver] # 断言条件:2025年1月20日之前(这里是美国时间)
3.postman发送请求

image-20240326154051767

4.Between Route Predicate 两个时间之间发送请求
1.需求分析

image-20240326154311538

2.应用实例
          predicates:- Path=/member/get/**- Between=2017-01-20T17:42:47.789-07:00[America/Denver], 2025-01-21T17:42:47.789-07:00[America/Denver] # 断言条件,在2017年1月20日到2025年1月21日之间
3.postman发送请求

image-20240326154819272

5.Cookie Route Predicate 携带某个Cookie的请求
1.需求分析

image-20240326154915495

2.应用实例
          predicates:- Path=/member/get/** - Cookie=user, sun # 断言条件,有cookie名为user,值为sun
3.postman发送请求并设置cookie
1.携带cookie

image-20240326155930608

2.不携带cookie

image-20240326155957204

6.Header Route Predicate 携带某个Header属性的请求
1.需求分析

image-20240326160231690

2.应用实例
          predicates:- Path=/member/get/**- Header=X-Request-Id, hello # 断言条件,Header中有X-Request-Id属性,值为hello
3.postman请求Header携带参数

image-20240326160636861

7.Host Route Predicate 指定主机发送的请求
1.需求分析

image-20240326160814835

2.应用实例
          predicates:- Path=/member/get/**- Host=**.sunxiansheng.com,**.anotherhost.org # 断言条件:任意前缀的sunxiansheng.com和anotherhost.org的host
3.postman请求修改host

image-20240326161932895

8.Method Route Predicate 指定请求方式的请求
1.应用实例
          predicates:- Path=/member/get/**- Method=GET,POST # 断言条件:请求方式为GET和POST
2.postman测试

image-20240326162427600

9.Path Route Predicate 指定路径的请求
应用实例
          predicates:- Path=/member/get/**, /member/get/save # 断言条件:路径为/member/get/**或者/member/get/save,其中的/是本服务的http://ip+端口/上下文路径/
10.Query Route Predicate 参数满足指定条件的请求
1.需求分析

image-20240326163249887

2.应用实例
          predicates:- Path=/member/get/**- Query=email, [\w-]+@([a-zA-Z]+\.)+[a-zA-Z]+ # 断言条件:请求携带参数为email,值要满足邮箱的格式
3.postman测试

image-20240326163835648

11.RemoteAddr Route Predicate 指定范围的ip客户端发送的请求
1.需求分析

image-20240326164333710

2.应用实例
          predicates:- Path=/member/get/**- RemoteAddr=127.0.0.1 # 断言条件:客户端的ip只能是本机
3.postman测试

image-20240326164931440

12.weight Route Predicate 根据权重进行断言

image-20240326165744615

7.过滤器

1.基本介绍
1.功能

image-20240326181720035

2.工作流程
  • 简单来说就是在远程调用之前和远程调用之后可以进行过滤
  • 过滤器链跟javaweb的过滤器一样
  • 原生的过滤器使用的并不多,更多的是使用自定义的过滤器

image-20240326181846252

2.自定义过滤器
1.需求分析

image-20240326184437570

2.代码实现 com/sun/springcloud/filter/CustomerGateWayFilter.java
package com.sun.springcloud.filter;import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;/*** Description: 编写全局过滤器** @Author sun* @Create 2024/3/26 18:50* @Version 1.0*/
@Component
public class CustomerGateWayFilter implements GlobalFilter, Ordered {/*** 在远程调用之前判断请求的参数user是否等于sun,pwd是否等于666** @param exchange the current server exchange* @param chain    provides a way to delegate to the next filter* @return*/@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {System.out.println("CustomerGateWayFilter被执行");// 获取请求的参数,由于使用get方法返回的是一个list所以需要get(0)String user = exchange.getRequest().getQueryParams().get("user").get(0);// 使用getFirst相当于获取了请求参数名为pwd的第一个元素String pwd = exchange.getRequest().getQueryParams().getFirst("pwd");// 不满足要求就设置状态码之后直接返回if (!("sun".equals(user) && "666".equals(pwd))) {System.out.println("非法用户");// 设置响应的状态码exchange.getResponse().setStatusCode(HttpStatus.NOT_ACCEPTABLE);// 返回return exchange.getResponse().setComplete();}System.out.println("放行");// 满足要求就放行return chain.filter(exchange);}/*** 根据返回的值来决定过滤器调用的顺序,数字越小则优先级越高** @return*/@Overridepublic int getOrder() {return 0;}
}
3.postman测试

image-20240326191515476

2

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Nacos部署(一)Linux部署Nacos2.3.x单机环境

&#x1f60a; 作者&#xff1a; 一恍过去 &#x1f496; 主页&#xff1a; https://blog.csdn.net/zhuocailing3390 &#x1f38a; 社区&#xff1a; Java技术栈交流 &#x1f389; 主题&#xff1a; Nacos部署&#xff08;一&#xff09;Linux部署Nacos2.3.x单机环境 ⏱️…

是德科技KEYSIGHT E4990A阻抗分析仪

181/2461/8938产品概述: E4990A 阻抗分析仪具有 20 Hz 至 120 MHz 的频率范围&#xff0c;可在宽阻抗范围内提供出色的 0.045%&#xff08;典型值&#xff09;基本准确度&#xff0c;并内置 40 V 直流偏置源&#xff0c;适用于元器件、半导体和材料测量。 无论研发、生产、质…

第四百二十五回

文章目录 1. 概念介绍2. 实现方法3. 示例代码4. 内容总结 我们在上一章回中介绍了"使用intl插件时遇到的问题"相关的内容&#xff0c;本章回中将介绍实现splash页面的另外一种方法.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 概念介绍 我们在本章回中介绍的…

YOLOv9改进策略:卷积魔改 | SPD-Conv,低分辨率图像和小物体涨点明显

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文改进内容&#xff1a;SPD-Conv由一个空间到深度(SPD)层和一个无卷积步长(Conv)层组成,特别是在处理低分辨率图像和小物体等更困难的任务时。 &#x1f4a1;&#x1f4a1;&#x1f4a1;SPD-Conv在多个数据集验证能够暴力涨点&#x…

网工内推 | 松下电器,解决方案工程师,最高25K,IE认证优先

01 松下电器 招聘岗位&#xff1a;基盘解决方案架构师 职责描述&#xff1a; 1、网络的规划设计&#xff0c;架构实施和故障排渣以及调优 2、负责网络设备的选型、搭建、系统监控、故障解决、性能优化 3、负责对集团内相关业务&#xff0c;进行提案或项目管理相关工作 4、对…

Vue3新手教程

Vue3新手教程 一. Vue3简介1. 性能的提升2.源码的升级3. 拥抱TypeScript4. 新的特性 二. 创建Vue3工程1. 基于 vue-cli 创建2. 基于 vite 创建(推荐)3. 一个简单的效果 三. Vue3核心语法1. OptionsAPI 与 CompositionAPI2. 拉开序幕的 setup2.1 setup 概述2.2 setup 的返回值2.…

自动化脚本-Excel批量生成二维码

演示&#xff1a; Excel二维码批量生成 exe资源文件&#xff1a; 下载exe 代码实现&#xff1a; pip install openpyxl openpyxl 是一个用于读写 Excel 文件的 Python 库 pip install qrcode qrcode 是一个 Python 库&#xff0c;可以用来生成二维码&#xff08;Quick Resp…

小目标检测篇 | YOLOv8改进之GSConv + Slim Neck提升小目标检测效果

前言:Hello大家好,我是小哥谈。在文章中,作者提出了一种新方法GSConv来减轻模型的复杂度并保持准确性。GSConv可以更好地平衡模型的准确性和速度。并且,提供了一种设计范式Slim Neck,以实现检测器更高的计算成本效益。实验过程中,与原始网络相比,改进方法获得了最优秀的…

【MySQL】锁

一、并发事务访问相同记录的三种情况 读-读情况   并发事务相继读取相同的记录&#xff0c;不会产生什么问题。 写-写情况   这种情况下会发生脏写的问题。 读-写或写-读情况   会发生脏读、不可重复读、幻读的情况 并发问题的解决方案 读操作利用多版本并发控制&am…

java的ArrayList类

ArrayList<E>E是自定义数据类型 ArrayList类&#xff1a; 构造函数&#xff1a; 成员方法&#xff1a; public boolean add(E e)&#xff1a; 将指定元素加到集合末尾 Appends the specified element to the end of this list. public class Array {public static…

动听的洗牌游戏(Java篇ArrayList实操)

本篇会加入个人的所谓‘鱼式疯言’ ❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言 而是理解过并总结出来通俗易懂的大白话, 小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的. &#x1f92d;&#x1f92d;&#x1f92d;可能说的不是那么严谨.但小编初心是能让更多人…

LeetCode - 存在重复元素

219. 存在重复元素 II 这道题可以用两个方法解决。 哈希表 从左到右遍历数组&#xff0c;并将数组的下标存到hash中&#xff0c;在遍历数字的过程中&#xff0c;如果hash中不存在nums[i]&#xff0c;将nums[i]加入到hash当中&#xff0c;若存在&#xff0c;则判断下标之间的关…

C#事件实例详解

一、什么是事件&#xff1f; 在C#中,事件(event)是一种特殊的类成员,它允许类或对象通知其他类或对象发生了某些事情。 从语法上看,事件的声明类似于字段,但它们在功能和行为上有一些重要的区别。 从技术角度来说,事件实际上是一个封装了事件订阅和取消订阅功能的委托字段。…

Python中的数据类型有四类八种如何理解?

在Python中&#xff0c;数据类型大致可以分为四大类&#xff0c;包含了八种基本的数据类型&#xff0c;这些分类有助于理解和使用Python进行编程。这四大类分别是&#xff1a; 数字类型 (Numeric Types): 整型 (int): 表示没有小数部分的整数&#xff0c;可以是正数、负数或零。…

海外媒体发稿:9种高效的媒体套餐内容发稿策略分析-华媒舍

海外媒体发稿&#xff1a;9种高效的媒体套餐内容发稿策略分析高效的媒体发布和营销推广策略对公司、本人的成就尤为重要。下面我们就对于媒体套餐内容发稿营销推广策略开展全面解析&#xff0c;帮助读者掌握并应用这9种合理的思路&#xff0c;进而获得更好的媒体营销效果。 1.媒…

Retelling|Facebook2

录音 Facebook 2 Retelling|Facebook2 复述转写 Hi, Im Helen Campbell, from DJ interpretation, European Commission, Im going to talk about Facebook. You Im sure that you are more familiar with Facebook, a lot, a lot more familiar than I than me. But Ive read…

C语言例4-19:求一元二次方程的解

求一元二次方程的解。 代码如下&#xff1a; //求一元二次方程的解 #include<stdio.h> #include<math.h> int main(void) {float a,b,c,d,x1,x2,p,q;printf("a,b,c?\n");scanf("%f,%f,%f",&a,&b,&c);printf("方程 ");…

智能楼宇3D可视化解决方案

什么是智能楼宇? 智能楼宇是为提高楼宇的使用合理性与效率,配置合适的建筑环境系统与楼宇自动化系统、办公自动化与管理信息系统以及先进的通信系统,并通过结构化综合布线系统集成为智能化系统的大楼。 面临的问题 信息孤岛,无法统一管理 各个子系统独立工作、独立管理,…

每天上万简历,录取不到1%!阿里腾讯的 offer 都给了哪些人?

三月天杨柳醉春烟~正是求职好时节~ 与去年秋招的冷淡不同&#xff0c;今年春招市场放宽了许多&#xff0c;不少企业纷纷抛出橄榄枝&#xff0c;各大厂的只差把“缺人”两个字写在脸上了。 字节跳动技术方向开放数10个类型岗位&#xff0c;研发需求占比60%&#xff0c;非研发新增…

redis关联和非关联

1.1.2.关联和非关联 传统数据库的表与表之间往往存在关联&#xff0c;例如外键&#xff1a; 而非关系型数据库不存在关联关系&#xff0c;要维护关系要么靠代码中的业务逻辑&#xff0c;要么靠数据之间的耦合&#xff1a; {id: 1,name: "张三",orders: [{id: 1,ite…