mysql数据同步ES方案---Canal

news2025/1/21 3:57:30

引言

之前公司开发社交APP的时候 在开发和初上线阶段,我们一直采用的是 MySQL 来存储用户的各种数据,满足基本的查询需求。当时系统业务量小,数据规模有限,因此 MySQL 能很好地支持查询操作,响应速度快,系统压力也较小。然而,随着产品推广和广告投入的增加,用户量迅速增长,查询需求变得更加复杂,这时单纯依靠 MySQL 支持所有查询就逐渐暴露出一些问题:

1. 数据量的增长
随着用户数增加、交易记录增多、日志信息积累,MySQL 数据表的规模逐步扩大。随着数据表变大,每次查询需要更多的时间遍历、扫描和过滤大量的数据行,导致查询响应速度变慢。

2. 查询需求的精细化
随着各种各样的需求上线,查询需求逐渐从简单筛选变为多条件筛选、模糊匹配、排序和聚合分析等更复杂的需求。MySQL 对于复杂查询的支持有限,即使通过优化索引,面对复杂的筛选和分析需求,查询效率仍无法满足。

3. 系统性能和压力问题
随着频繁的复杂查询增多,数据库的负载也随之增加。尤其是在数据量和并发访问量增加的情况下,MySQL 的性能瓶颈愈加明显,直接影响到系统的整体响应速度,甚至会导致响应变慢或短时宕机。

为了解决这些问题,我们采用了“空间换效率”的策略,引入了 Elasticsearch。通过将用户行为数据从 MySQL 异步同步到 Elasticsearch,我们利用其强大的搜索、聚合和分析能力,将这些复杂查询从 MySQL 转移到 Elasticsearch。这种方案不仅大大减轻了 MySQL 的查询负担,也使系统能够在海量数据环境下保持快速响应,从而更好地支持社交产品中的实时和复杂查询需求。

为了实现 MySQL 数据到 Elasticsearch 的实时同步,我们面临的核心问题是如何将 MySQL 数据变化高效、准确地推送到 Elasticsearch。理想的方案应该具备以下几个特点:

  1. 实时性:确保 MySQL 数据的增删改操作能够迅速反映在 Elasticsearch 中,使查询结果始终保持最新状态。
  2. 高效性:避免每次数据更新都进行全量同步,应该仅将变化部分同步,减少资源消耗。
  3. 数据一致性:确保 MySQL 和 Elasticsearch 中的数据保持一致,避免出现因延迟或丢失导致的数据不一致问题。

为了解决这一系列问题,常见的数据同步方案包括同步双写、异步双写、基于 SQL 的数据抽取和基于 Binlog 的实时同步等。这些方案各有优缺点,且在不同场景下会选择不同的工具。例如,业界常用的同步工具有 Canal、DTS、Databus、Flink、CloudCanal、Maxwell、DRD 以及 Yugong 等。

今天我将重点介绍其中一种高效的实时同步工具——Canal

Canal 简介

Canal 是由阿里巴巴开源的一款数据捕获工具,专为解决 MySQL 数据实时同步问题设计。它通过模拟 MySQL 从库的方式来监听和解析 binlog 日志,从而获取 MySQL 的数据变更,支持将这些变更同步到其他数据存储系统,比如 Redis、Elasticsearch、Kafka 等,满足多种数据同步需求。

Canal 的同步原理

Canal 的工作原理主要基于 MySQL binlog 日志。MySQL binlog 是一个记录所有数据更改操作(INSERT、UPDATE、DELETE)的日志文件,通常用于数据库的主从同步。Canal 利用 binlog 来实现数据同步,其核心原理包括以下几个步骤:

  1. 模拟 MySQL 从库
    Canal 通过伪装成 MySQL 的从库来连接到主库,就像主从复制一样读取 binlog 日志,抓取所有的数据变更事件。这种方式无需对主库进行代码改动,且不影响数据库性能。
  2. 解析 binlog 日志
    Canal 从 binlog 中获取的数据是二进制格式,需要对其进行解析。Canal 可以解析常见的增删改操作(如 INSERT、UPDATE、DELETE),并将这些操作解析成可以被识别的 JSON 格式,便于后续处理。
  3. 事件发布与同步
    解析完成后,Canal 将这些数据变更事件推送给指定的目标系统,比如 Kafka 或直接写入 Elasticsearch。这样,每当 MySQL 中的数据发生变更时,Elasticsearch 就能及时收到并更新,保持与 MySQL 数据的一致性。

这种方式实现了数据的实时性和准确性,且具备较好的扩展性,不会增加 MySQL 主库的负载。Canal 的这一同步机制,让它成为高效、稳定的数据同步工具,广泛应用于实时数据同步和分布式数据系统中。

方案选型

在数据同步过程中,我们可以直接使用 Canal 自带的 Canal Adapter,将 MySQL 数据变化实时同步到 Elasticsearch。这种方式实现简单且直观,适合对同步性能要求较高且系统结构相对简单的场景。

然而,在实际应用中,业务需求往往更为复杂,我们可能需要更加灵活的同步方式,比如异步处理多系统分发、或者更高的容错能力。在这些情况下,我们可以在 Canal 和 Elasticsearch 之间引入消息队列(如 RabbitMQ),以满足以下场景需求:

  • 多系统分发
    如果数据不仅需要同步到 Elasticsearch,还需同步到其他系统(如缓存系统、推荐系统等),引入 RabbitMQ 可以让 Canal 只负责将数据推送到 RabbitMQ,由多个消费端从 RabbitMQ 订阅消息并写入不同的目标系统,实现灵活的数据分发。
  • 异步与容错处理
    Canal 和 Elasticsearch 的直接连接是实时且同步的,而 RabbitMQ 可以作为缓冲区来存储 Canal 推送的变更事件,便于异步消费。如果 Elasticsearch 出现短暂故障,RabbitMQ 会将 Canal 的消息缓存,待 Elasticsearch 恢复后再继续消费,避免数据丢失。
  • 负载均衡
    通过 RabbitMQ,还可以实现同步负载的分摊,防止 Canal 和 Elasticsearch 直接连接时出现高负载问题,有效保障系统的稳定性。

因此,在数据同步方案设计中,如果对系统的稳定性、容错性和可扩展性有更高要求,可以考虑使用 RabbitMQ 等消息队列来增强 Canal 的同步能力,实现灵活而高效的数据同步。

架构设计

基于上述方案,我将使用 Hyperf 框架,结合 CanalRabbitMQMySQLElasticsearch,实现一个简单而灵活的数据同步功能。具体来说,通过 Canal 监听 MySQL 的 binlog 日志,将数据变更事件推送到 RabbitMQ 中,Hyperf 作为消费端从 RabbitMQ 接收这些变更事件,处理后将数据同步到 Elasticsearch 中。此架构保证了系统的高容错性和扩展性,同时实现了高效的异步数据同步。

备注:系统所需的 RabbitMQ、MySQL 和 Elasticsearch 环境已通过 Docker 安装配置。

使用 Docker 安装和配置 Canal 服务端(含 RabbitMQ 和 MySQL 配置)

  • 创建挂载目录
    在主机上创建一个挂载目录,以便持久化 Canal 的配置文件和日志文件:

    mkdir -p /docker/canal/conf
    mkdir -p /docker/canal/logs
    
  • 首次启动 Canal 容器
    启动 Canal 容器,让它生成默认的配置文件:

    docker run --name canal -d canal/canal-server:latest
    
  • 复制默认配置文件到挂载目录
    canal.propertiesinstance.properties 配置文件从容器复制到主机挂载目录,以便后续配置:

    docker cp canal:/home/admin/canal-server/conf/canal.properties /docker/canal/conf/
    docker cp canal:/home/admin/canal-server/conf/example/instance.properties /docker/canal/conf/
    
  • 停止并删除容器
    复制完成后,停止并删除 Canal 容器:

    docker stop canal
    docker rm canal
    
  • 编辑 canal.properties 以配置 RabbitMQ
    打开挂载目录下的 canal.properties 文件,添加 RabbitMQ 的相关配置。以下是示例配置:

    在这里插入图片描述

在这里插入图片描述

```properties
canal.serverMode = rabbitMQ
rabbitmq.host = 192.168.110.190:5672
rabbitmq.virtual.host = /
rabbitmq.exchange = canal_exchange
rabbitmq.username = admin
rabbitmq.password = admin
rabbitmq.queue = 
rabbitmq.routingKey = 
rabbitmq.deliveryMode = 2
```

*   **`canal.serverMode`**:用于指定 Canal 的消息投递模式,支持以下几种模式:TCP、Kafka、RocketMQ、RabbitMQ 和 PulsarMQ。在这里,设置为 `rabbitMQ`,意味着 Canal 将 MySQL 数据变化通过 RabbitMQ 推送,供后续的消费端处理。
*   **`rabbitmq.host`**:RabbitMQ 的服务地址和端口。在这里,`192.168.110.190:5672` 表示 RabbitMQ 服务器的 IP 地址是 `192.168.110.190`,端口号为 `5672`。
*   **`rabbitmq.virtual.host`**:指定 RabbitMQ 的虚拟主机(Virtual Host),这是 RabbitMQ 中的一个逻辑隔离环境。`/` 表示使用默认虚拟主机。
*   **`rabbitmq.exchange`**:Canal 数据推送所使用的交换机名称。这里指定为 `canal_exchange`,用于在 RabbitMQ 中接收 Canal 的数据变更消息。交换机会根据消息的 routing key 将消息路由到相应的队列。
*   **`rabbitmq.username` 和 `rabbitmq.password`**:RabbitMQ 的用户名和密码。
*   **`rabbitmq.queue`**:指定 Canal 推送数据的目标队列名称。消费者可以从这个队列中接收 MySQL 数据变更事件。
*   **`rabbitmq.routingKey`**:用于消息路由的路由键。通常可以指定为数据库名称和表名(例如 `${db}.${table}`),以便将消息发送到正确的队列。
*   **`rabbitmq.deliveryMode`**:指定消息的持久化方式。设置为 `2` 表示消息将被持久化到磁盘,以确保消息在 RabbitMQ 重启或崩溃时不会丢失。
  • 编辑 instance.properties 以配置 MySQL
    在挂载目录下找到 instance.properties 文件,填写 MySQL 的连接信息,配置示例如下:

    在这里插入图片描述

    在这里插入图片描述

    properties
    复制代码
    canal.instance.master.address=<MYSQL_HOST>:<MYSQL_PORT>
    canal.instance.dbUsername=<MYSQL_USERNAME>
    canal.instance.dbPassword=<MYSQL_PASSWORD>
    canal.instance.filter.regex=<DATABASE_NAME>.<TABLE_NAME>
    
    • canal.instance.master.address:MySQL 主库的 IP 和端口。
    • canal.instance.dbUsernamecanal.instance.dbPassword:连接 MySQL 的用户名和密码,确保该用户有权限读取 binlog。
    • canal.instance.filter.regex:指定要同步的数据库和表,例如 circle.circle_table,如果不设置此项,将同步所有表。

重新启动 Canal 容器并挂载配置文件
重新启动 Canal 容器,将配置文件和日志目录挂载到容器中:

docker run -d --name canal \
    -p 11111:11111 \
    -v /docker/canal/conf:/home/admin/canal-server/conf \
    -v /docker/canal/logs:/home/admin/canal-server/logs \
    canal/canal-server:latest

这样,Canal 服务端已通过 Docker 安装并完成 RabbitMQ 和 MySQL 的配置。现在可以将 MySQL 数据变更事件推送至 RabbitMQ,接下来可以使用消费端(如 Hyperf 框架)来接收并同步数据到 Elasticsearch。

技术流程实现

步骤 1:创建数据库和表

首先,我们需要在 MySQL 中创建一个名为 circle 的数据库,以及一个表 circle_global,用于存储圈子相关的数据。

1.1 创建数据库

在 MySQL 中执行以下 SQL 语句,创建数据库:

CREATE DATABASE circle;
1.2 创建表 circle_global

接下来,切换到刚创建的数据库,并创建名为 circle_global 的表。执行以下 SQL 语句:

USE circle;

CREATE TABLE `circle_global` (
  `id` bigint(20) NOT NULL AUTO_INCREMENT,
  `circle_id` bigint(20) NOT NULL,
  `user_id` bigint(20) NOT NULL,
  `created_at` datetime NOT NULL,
  `like_count` int(11) DEFAULT '0',
  PRIMARY KEY (`id`),
  KEY `idx_created_at` (`created_at`),
  KEY `idx_like_count` (`like_count`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
1.3 表结构说明
  • id:主键,唯一标识每条记录,采用自增长方式。
  • circle_id:圈子的 ID,用于标识不同的圈子。
  • user_id:用户的 ID,标识参与该圈子的用户。
  • created_at:记录创建的时间,采用 datetime 类型。
  • like_count:记录该圈子的点赞数,默认值为 0。

表的索引:

  • 主键索引:对 id 列进行主键索引,确保每条记录的唯一性。
  • 创建时间索引:在 created_at 列上创建索引,以加速基于时间的查询。
  • 点赞数索引:在 like_count 列上创建索引,以支持点赞数的快速统计和查询。

步骤 2:在 Elasticsearch 中创建索引

在 MySQL 数据库和表创建完成后,我们需要在 Elasticsearch 中创建一个索引,以便存储和管理从 circle_global 表中同步过来的数据。

2.1 创建索引 circle_global

使用以下 curl 命令在 Elasticsearch 中创建索引 circle_global,并定义索引的映射:

curl -X PUT "http://localhost:9200/circle_global?pretty" -H 'Content-Type: application/json' -d'
{
  "mappings": {
    "properties": {
      "circle_id": {
        "type": "long"
      },
      "user_id": {
        "type": "long"
      },
      "created_at": {
        "type": "date"
      },
      "like_count": {
        "type": "integer"
      }
    }
  }
}
'
2.2 映射说明
  • circle_id:类型为 long,用于存储圈子 ID。
  • user_id:类型为 long,用于存储用户 ID。
  • created_at:类型为 date,用于存储创建时间。
  • like_count:类型为 integer,用于存储点赞数。
2.3 验证索引创建

创建索引后,可以使用以下命令验证索引是否成功创建:

curl -X GET "http://localhost:9200/_cat/indices?v"

在返回的列表中查找 circle_global 索引,如果存在,表示索引创建成功。

步骤 3:在 Hyperf 中安装 AMQP 和 Elasticsearch 组件

在准备好 MySQL 和 Elasticsearch 之后,我们需要在 Hyperf 项目中安装必要的组件,以便能够与 RabbitMQ 和 Elasticsearch 进行交互。

3.1 安装 AMQP 组件

使用 Composer 安装 Hyperf 的 AMQP 组件,执行以下命令:

composer require hyperf/amqp
3.2 安装 Elasticsearch 组件

同样,使用 Composer 安装 Elasticsearch 的 PHP 客户端,执行以下命令:

composer require elasticsearch/elasticsearch
3.3 配置环境变量

在项目根目录下找到 .env 文件,并添加以下配置,以便 AMQP 和 Elasticsearch 组件能够正确连接到相应的服务。

APP_NAME=skeleton
APP_ENV=dev

# 数据库 配置
DB_DRIVER=mysql
DB_HOST=192.168.110.190
DB_PORT=3306
DB_DATABASE=circle
DB_USERNAME=root
DB_PASSWORD=123456
DB_CHARSET=utf8mb4
DB_COLLATION=utf8mb4_unicode_ci
DB_PREFIX=

# RabbitMQ 配置
AMQP_HOST=192.168.110.190
AMQP_PORT=5672
AMQP_USER=admin
AMQP_PASSWORD=admin
AMQP_VHOST=/
AMQP_ENABLE=true

# Elasticsearch 配置
ELASTICSEARCH_HOST=192.168.110.190:9200
3.4 更新 AMQP 配置文件

config/autoload/amqp.php 文件中,确保 AMQP 的连接信息与 .env 中的环境变量一致:

return [
    'default' => [
        'host' => env('RABBITMQ_HOST', 'localhost'),
        'port' => env('RABBITMQ_PORT', 5672),
        'user' => env('RABBITMQ_USER', 'guest'),
        'password' => env('RABBITMQ_PASSWORD', 'guest'),
        'vhost' => env('RABBITMQ_VHOST', '/'),
    ],
];
3.5 更新 Elasticsearch 配置文件

config/autoload/elasticsearch.php 文件中,确保 Elasticsearch 的连接信息与 .env 中的环境变量一致:

return [
    'default' => [
        'hosts' => [
            env('ELASTICSEARCH_HOST', 'localhost:9200'),
        ],
    ],
];

步骤 4:使用单例模式创建 Elasticsearch 实例

为了避免在应用程序中重复创建 Elasticsearch 客户端实例,我们可以使用单例模式来管理 Elasticsearch 的连接。这种方式确保在整个应用生命周期内,只创建一次 Elasticsearch 客户端,从而提高资源利用率并减少开销。

4.1 创建 Elasticsearch 单例类

app/Components 目录下创建一个名为 Elasticsearch.php 的类,用于管理 Elasticsearch 客户端的单例实例:

<?php

namespace App\Components;

use Elasticsearch\ClientBuilder;

class Elasticsearch
{
    private static $instance;

    /**
     * 私有化构造函数,防止外部实例化
     */
    private function __construct()
    {
    }

    /**
     * 禁止克隆对象
     */
    private function __clone()
    {
    }

    /**
     * 获取 Elasticsearch 客户端的单例实例
     *
     * @return \Elasticsearch\Client
     */
    public static function getInstance()
    {
        if (is_null(self::$instance)) {
            self::$instance = ClientBuilder::create()
                ->setHosts([env('ELASTICSEARCH_HOST', 'localhost:9200')])
                ->build();
        }

        return self::$instance;
    }
}
4.2 代码说明
  • 私有化构造函数:通过私有化构造函数,防止外部代码实例化该类。
  • 单例获取方法getInstance 方法用于获取 Elasticsearch 客户端的单例实例。如果实例尚未创建,则调用 ClientBuilder 创建新的实例,并将其存储在 self::$instance 中。
  • 防止克隆:通过私有化 __clone 方法,确保单例实例不能被克隆。

通过这种设计,我们能够在 Hyperf 应用中高效地管理 Elasticsearch 客户端的连接,避免了资源浪费和连接创建的开销。接下来,您可以在需要使用 Elasticsearch 的地方直接调用 Elasticsearch::getInstance() 来获取客户端实例。

步骤 5:创建消费模式并实现监听 Canal 的逻辑

在 Hyperf 中,我们将创建一个消费者类,监听 Canal 发送的消息,并将这些消息写入 Elasticsearch 实例中。以下是实现的详细步骤:

5.1 创建 Canal 消费者类

app/Amqp/Consumer 目录下创建一个名为 CanalConsumer.php 的类,用于处理 Canal 的消息:

<?php

namespace App\Amqp\Consumer;

use Hyperf\Amqp\Message\ConsumerMessage;
use Hyperf\Amqp\Result;
use App\Components\Elasticsearch;
use Hyperf\Amqp\Annotation\Consumer;
use Hyperf\Amqp\Annotation\Inject;

#[Consumer(exchange: 'canal_exchange', routingKey: 'canal', queue: 'canal_queue', name: "CanalConsumer", nums: 1)]
class CanalConsumer extends ConsumerMessage
{
    public function consume($data): string
    {
        // 解析 Canal 发送的数据
        $canalMessage = json_decode($data, true);

        // 检查是否包含数据库变更信息
        if (isset($canalMessage['data']) && is_array($canalMessage['data'])) {
            foreach ($canalMessage['data'] as $row) {
                // 获取 Elasticsearch 客户端实例
                $client = Elasticsearch::getInstance();

                // 准备要写入 Elasticsearch 的参数
                $params = [
                    'index' => 'circle_global', // Elasticsearch 的索引名称
                    'id' => $row['id'],         // 使用 MySQL 的 id 作为文档 ID
                    'body' => [
                        'circle_id' => $row['circle_id'],
                        'user_id' => $row['user_id'],
                        'created_at' => $row['created_at'],
                        'like_count' => $row['like_count'],
                    ],
                ];

                // 将数据写入到 Elasticsearch 索引中
                $client->index($params);
            }
        }

        return Result::ACK; // 确认消息消费成功
    }
}
5.2 代码说明
  • 注解配置:使用 #[Consumer] 注解配置消费者的信息,指定交换机、路由键、队列名和并发消费数量。
  • 解析消息:在 consume 方法中,解析 Canal 发送的 JSON 数据。确保数据存在并为数组格式。
  • 循环处理数据:如果 data 字段存在,循环处理每一行数据。
  • 获取 Elasticsearch 实例:调用 Elasticsearch::getInstance() 获取客户端实例,避免重复创建。
  • 准备 Elasticsearch 写入参数:将从 Canal 获取的数据映射到 Elasticsearch 索引的字段。
  • 写入 Elasticsearch:使用 $client->index($params) 将数据写入指定的 Elasticsearch 索引。
  • 返回确认结果:返回 Result::ACK,确认消息已成功消费。
5.3 启动 Hyperf 服务

确保所有配置和代码都已完成后,启动 Hyperf 服务:

php bin/hyperf.php start

在这里插入图片描述

可以看到 CanalConsumer消费者已经成功启动监听

5.4 测试数据同步

  1. 在 MySQL 的 circle_global 表中执行插入操作。

  2. 确认 Canal 正常工作,并查看 CanalConsumer 是否成功接收到消息并写入到 Elasticsearch。

  3. 使用以下命令查询 Elasticsearch,确认数据是否已经同步:

    curl -X GET "http://localhost:9200/circle_global/_search?pretty"
    

    在这里插入图片描述

可以看到,我们已经成功实现了通过监听 Canal 发送的消息,将 MySQL 中的变更数据实时写入 Elasticsearch 实例。这一过程不仅提高了数据处理的效率,也为后续的数据分析和搜索提供了强有力的支持。

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

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

相关文章

你认为BI不需要建模,那就大错特错了

BI 是一种数据类的技术解决方案&#xff0c;将许多来自不同企业业务系统的数据提取有分析价值的数据进行清洗、转换和加载&#xff0c;就是抽取Extraction、转换 Transformation、加载Loading 的ETL过程&#xff0c;最终合并到一个数据仓库中&#xff0c;按照一定的建模方式例如…

【学习】安装cudf和cuml

为了把cpu上跑的SVM程序搬到GPU上跑&#xff0c;需要装这俩包&#xff0c;但是搞了半天装不上&#xff0c;最后发现是清华源的问题。换了中科大的源没问题了。 rapids官网&#xff1a;RAPIDS | GPU Accelerated Data Science 官网安装&#xff1a;Installation Guide - RAPID…

安装CentOS 8镜像和创建CentOS 8虚拟机教程

一、安装虚拟机 网上查找教程&#xff0c;我用的是VMware 17 二、下载CentOS 8镜像 1.阿里云下载CentOS 8镜像 centos安装包下载_开源镜像站-阿里云 (aliyun.com) 选择需要下载的版本&#xff0c;(建议)下载dvd1版本的iso&#xff08;也有下载boot版本的iso&#xff0c;创…

用柔性神经k-Opt学习搜索路径问题的可行和不可行区域(未完,先看前驱文章L2S)

文章目录 Abstract1 IntroductionAbstract 介绍了一种名为 Neural k-Opt(NeuOpt)的新型学习搜索(L2S)求解器,用于解决路径问题。它学习执行基于定制的动作分解方法和定制的循环双流(Recurrent Dual-Stream)解码器的灵活 k-opt 交换。 作为一项开创性的工作,我们绕过了…

使用YOLOv8进行实时人员跟踪和计数

在计算机视觉领域,实时跟踪和统计人数对于各种应用至关重要,从监控到事件管理。在这篇博客文章中,我们将探讨如何利用YOLOv8和ByteTracker实现准确的人数统计。 引言 YOLOv8(You Only Look Once,第八版)是一种以其速度和准确性而闻名的最新对象检测模型。 ByteTracker是一…

Oracle DECODE 丢失时间精度的原因与解决方案

在Oracle数据库中&#xff0c;DECODE 函数是一个非常实用的条件处理函数&#xff0c;通常用于替代简单的 CASE WHEN 语句。它根据给定的值列表进行匹配&#xff0c;如果匹配成功则返回相应的值。如果不匹配&#xff0c;返回一个默认值。 问题描述 SELECT DECODE(-21, -1, NU…

低代码可视化-uniapp购物车页面-代码生成器

购物车页面是电子商务网站或应用程序中的一个关键功能页面&#xff0c;它允许用户查看、编辑和管理他们选择加入购物车的商品。下面通过低代码可视化实现一个uniapp购物车页面&#xff0c;把购物车整个事件都集成进去。实现完成后可以保存为页面模板。 收货地址选择 如果尚未…

基于SpringBoot+Vue+uniapp微信小程序的垃圾分类系统的详细设计和实现(源码+lw+部署文档+讲解等)

项目运行截图 技术框架 后端采用SpringBoot框架 Spring Boot 是一个用于快速开发基于 Spring 框架的应用程序的开源框架。它采用约定大于配置的理念&#xff0c;提供了一套默认的配置&#xff0c;让开发者可以更专注于业务逻辑而不是配置文件。Spring Boot 通过自动化配置和约…

Cursor 、v0 和 Bolt.new:当今 AI 编程工具的全面解析与对比

AI 驱动的编码工具正在迅速发展&#xff0c;每次新版本的发布都承诺将改变我们的开发流程。这些工具旨在通过智能代码建议、重复任务的自动化&#xff0c;甚至从自然语言提示生成整个代码片段来提升生产力。10月4日&#xff0c;StackBlitz 推出了 Bolt.new&#xff0c;进一步丰…

建议大家尽早培养自己的表达能力!分享6个有效工具~

在日常工作生活中&#xff0c;有效沟通是成事的不二法则。现在很多人不热衷沟通&#xff0c;对人际交往没热情&#xff0c;但是会和不用&#xff0c;与不想也不会是两回事&#xff0c;所以掌握一些沟通技巧很有必要。 如何进行有效沟通&#xff1f;推荐大家进行结构化表达和沟…

代码复现(五):GCPANet

文章目录 net.py1.class Bottleneck&#xff1a;残差块2.class ResNet&#xff1a;特征提取3.class SRM&#xff1a;SR模块4.class FAM&#xff1a;FIA模块5.class CA&#xff1a;GCF模块6.class SA&#xff1a;HA模块7.class GCPANet&#xff1a;网络架构 train.pytest.py 论文…

计算机视觉——人像的分割与无缝融合

1.概述 新加坡现代汽车集团创新中心的一篇新论文提供了一种在计算机视觉中分离“融合”人像的方法——在这些情况下&#xff0c;对象识别框架发现一个人在某种程度上与另一个人“太接近”&#xff08;例如例如“拥抱”动作或“站在后面”的姿势&#xff09;&#xff0c;并且无法…

从零实现数据结构:堆的实现和简单堆排序

事先说明&#xff0c;这里采用的都是小堆。下面是代码中的小堆示意图 这里向大家分享一个常见数据结构可视化的网址&#xff1a;Data Structure Visualization (usfca.edu) 声明部分heap.h&#xff1a; #define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include &…

Turtle画树丛

Turtle画树丛&#xff0c;50棵树&#xff0c;左侧的树向左倾斜&#xff0c;右侧的树向右倾斜。 完整代码如下&#xff1a; import turtle import randomdef draw_tree(pos,hd,angle,len,init_len,level): t.penup()t.goto(pos)t.pendown()t.setheading(hd)if pos[0]<0:t…

基于C++实现(控制台)职工信息管理系统

高级程序设计实验报告 一、实验内容 设计一个职工信息管理案例&#xff0c;实现不同类别职工的工资数据。职工的种类包括&#xff1a;正式职工和临时工。定义一个基本的职工类&#xff0c;基本信息包括&#xff1a;编号、姓名、性别、年龄、家庭住址、基本职务工资。派生出正…

当前python文件所在位置的上级文件夹的路径表示法:..的等价表示法os.pardir

【小白从小学Python、C、Java】 【考研初试复试毕业设计】 【Python基础AI数据分析】 当前python文件 所在位置的上级文件夹 的路径表示法&#xff1a;.. 的等价表示法 os.pardir [太阳]选择题 下列说法中正确的是? import os parent_dir os.pardir print("【显示】…

Axure重要元件三——中继器表单制作

亲爱的小伙伴&#xff0c;在您浏览之前&#xff0c;烦请关注一下&#xff0c;在此深表感谢&#xff01; 本节课&#xff1a;中继器表单制作 课程内容&#xff1a;利用中继器制作表单 应用场景&#xff1a;台账、表单 案例展示&#xff1a; 步骤一&#xff1a;建立一个背景区…

【WPF】04 Http消息处理类

这里引入微软官方提供的HttpClient类来实现我们的目的。 首先&#xff0c;介绍一下官方HttpClient类的内容。 HttpClient 类 定义 命名空间: System.Net.Http 程序集: System.Net.Http.dll Source: HttpClient.cs 提供一个类&#xff0c;用于从 URI 标识的资源发送 HTTP 请…

探索性数据分析 (EDA) 简介

文章目录 一、介绍二、探索性数据分析 (EDA)三、探索性数据分析的类型3.1 单变量分析3.2 双变量分析3.3 多元分析 四、理解 EDA五、结论 一、介绍 探索性数据分析是研究或理解数据并提取洞察数据集以识别数据模式或主要特征的过程。EDA 通常分为两种方法&#xff0c;即图形分析…

Spring Boot 整合 RocketMQ 之顺序消息

前言&#xff1a; 上一篇我们分享了 Spring Boot 整合 RocketMQ 完成普通消息发送的过程&#xff0c;本篇我们来分享一下 RocketMQ 顺序消息的发送。 RocketMQ 系列文章传送门 RocketMQ 的介绍及核心概念讲解 Spring Boot 整合 RocketMQ 之普通消息 Spring Boot 整合 Rock…