当前位置: 首页 > news >正文

RabbitMQ实现数据库与ElasticSearch的数据同步和分享文件过期处理

🎈 1 参考文档

RabbitMQ实现数据库与ElasticSearch的数据同步 | Hannya。-CSDN

企业级开发项目实战——基于RabbitMQ实现数据库、elasticsearch的数据同步 | 波总说先赚它一个小目标-CSDN

SPringBoot集成RabbitMQ实现30秒过期删除功能 | 军大君-CSDN


🔍 2 个人需求

  1. 当进行文件上传、文件创建、文件重命名等操作时:

    通过RabbitMQ:

    • 生产者:文件服务,执行上传、创建、重命名等文件操作,将用户文件信息(例如文件名、文件ID等)发送到RabbitMQ新增队列。
    • 消费者:查询服务,监听RabbitMQ新增队列,一旦收到消息,将用户文件信息新增或更新到Elasticsearch中。
  2. 文件删除时:

    通过RabbitMQ:

    • 生产者:文件服务,执行文件删除操作,将用户文件ID发送到RabbitMQ删除队列。
    • 消费者:查询服务,监听 RabbitMQ 队列,一旦收到消息,通过用户文件ID从Elasticsearch中删除相应的用户文件信息。
  3. 根据文件名进行文件模糊查询:

    通过OpenFeign:

    • 生产者:文件服务,查询服务调用文件服务提供的OpenFeign接口,通过用户文件ID从查询该用户文件是否存在。
    • 消费者:查询服务,如果不存在,将数据根据用户文件ID从Elasticsearch中删除。
  4. 分享文件时间到期处理:

    通过RabbitMQ的TTL(生存时间) + 死信队列:

    • 生产者:文件服务, 使用TTL模拟一个“延时队列”,在文件分享时间到期后,将消息传递到死信队列。
    • 消费者:文件服务,死信监听器监听到之后,将分享文件的分享状态改为已过期状态。

🔈3 声明

只是提供思路,代码不是很完整,直接复制运行不了。

最后面有完整网盘项目代码。


🚀4 OpenFeign相关部分(查询服务)

4.1 引入依赖

<!-- nacos -->
<dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>
<!-- openfeign -->
<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
<!-- loadbalancer -->
<dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-loadbalancer</artifactId>
</dependency>

4.2 application.yml

spring:# nacos注册的服务名application:name: netdisk-searchcloud:nacos:discovery:# 配置注册服务的IP地址server-addr: (IP地址):8848username: nacospassword: nacos

4.3 FileFeignService 接口

@FeignClient(name = "netdisk-file", configuration = FeignInterceptor.class)
public interface FileFeignService {@RequestMapping("/file/getUserFile/{userFileId}")ResultResponse<Boolean> getUserFile(@PathVariable Long userFileId);
}

4.4 @EnableFeignClients 注解

@ComponentScan(value = "com.cauli.search.*")
@EnableFeignClients(basePackages = "com.cauli.search")
@SpringBootApplication
public class NetdiskSearchApplication {public static void main(String[] args) {SpringApplication.run(NetdiskSearchApplication.class, args);}
}

🚀5 Elasticsearch相关部分(查询服务)

5.1 引入依赖

<!-- elasticsearch -->
<dependency><groupId>co.elastic.clients</groupId><artifactId>elasticsearch-java</artifactId><version>8.0.1</version>
</dependency>
<dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.12.3</version>
</dependency>
<dependency><groupId>jakarta.json</groupId><artifactId>jakarta.json-api</artifactId><version>2.0.1</version>
</dependency>

5.2 application.yml

# elasticsearch相关的配置
elasticsearch:# ES网关地址hostname: (IP地址)# ES网关端口port: 9200# ES网官方方案scheme: http

5.3 ElasticSearchConfig 配置类

@Configuration
public class ElasticSearchConfig {@Value("${elasticsearch.hostname}")String hostname;@Value("${elasticsearch.port}")int port;@Value("${elasticsearch.scheme}")String scheme;@Beanpublic ElasticsearchClient elasticsearchClient(){// 创建低级客户端RestClient client = RestClient.builder(new HttpHost(hostname, port,scheme)).build();// 创建API客户端,使用Jackson映射器创建传输层ElasticsearchTransport transport = new RestClientTransport(client,new JacksonJsonpMapper());return new ElasticsearchClient(transport);}
}	

5.4 Elasticsearch 服务类和服务实现类

public interface ElasticsearchService {/*** 更新ES数据** @param fileSearchDTO*/void uploadES(FileSearchDTO fileSearchDTO);/*** 删除ES数据** @param userFileId*/void deleteES(Long userFileId);/*** 搜索ES数据** @return*/List<SearchFileVO> searchES(SearchFileQueryDTO searchFileVO);
}
@Slf4j
@Service
public class ElasticsearchServiceImpl implements ElasticsearchService {@Autowiredprivate ElasticsearchClient elasticsearchClient;@Resourceprivate FileFeignService feignService;private final ThreadPoolExecutor executor = new ThreadPoolExecutor(12, // 核心线程数20, // 最大线程数1, // 线程存活时间TimeUnit.SECONDS, // 存活时间单位new ArrayBlockingQueue<>(1000) // 任务队列);public void uploadES(FileSearchDTO fileSearchDTO) {executor.execute(() -> {try {elasticsearchClient.index(i -> i.index("file_search").id(fileSearchDTO.getUserFileId()).document(fileSearchDTO));} catch (IOException e) {throw new RuntimeException(e);}});}public void deleteES(Long userFileId) {executor.execute(() -> {try {elasticsearchClient.delete(d -> d.index("file_search").id(String.valueOf(userFileId)));} catch (Exception e) {log.debug("ES删除操作失败,请检查配置");}});}@Overridepublic List<SearchFileVO> searchES(SearchFileQueryDTO searchFileQueryDTO) {int pageNum = (int) searchFileQueryDTO.getPageNum() - 1;int pageSize = (int) (searchFileQueryDTO.getPageSize() == 0 ? 10 : searchFileQueryDTO.getPageSize());SearchResponse<FileSearchDTO> search = null;try {search = elasticsearchClient.search(s -> s.index("file_search").query(_1 -> _1.bool(_2 -> _2.must(_3 -> _3.bool(_4 -> _4.should(_5 -> _5.match(_6 -> _6.field("fileName").query(searchFileQueryDTO.getFileName()))).should(_5 -> _5.wildcard(_6 -> _6.field("fileName").wildcard("*" + searchFileQueryDTO.getFileName() + "*"))))).must(_3 -> _3.term(_4 -> _4.field("userId").value(StpUtil.getLoginIdAsLong()))))).from(pageNum).size(pageSize).highlight(h -> h.fields("fileName", f -> f.type("plain").preTags("<span class='keyword'>").postTags("</span>")).encoder(HighlighterEncoder.Html)), FileSearchDTO.class);} catch (IOException e) {e.printStackTrace();}List<SearchFileVO> searchFileVOList = new ArrayList<>();if (search != null) {for (Hit<FileSearchDTO> hit : search.hits().hits()) {SearchFileVO searchFileVO = new SearchFileVO();BeanUtil.copyProperties(hit.source(), searchFileVO);searchFileVO.setHighLight(hit.highlight());searchFileVOList.add(searchFileVO);// 如果文件不存在,也从ES中删除if (!feignService.getUserFile(searchFileVO.getUserFileId()).getData()) {executor.execute(() -> this.deleteES(searchFileVO.getUserFileId()));}}}return searchFileVOList;}
}

5.5 ElasticsearchController 前端控制器

@RestController
@RequestMapping("/search")
public class ElasticsearchController {@Autowiredprivate ElasticsearchService elasticService;@GetMapping(value = "/searchFile")public RestResult<SearchFileVO> searchFile(SearchFileQueryDTO searchFileQueryDTO) {List<SearchFileVO> searchFileVOList = elasticService.searchES(searchFileQueryDTO);return RestResult.success().dataList(searchFileVOList, searchFileVOList.size());}
}

5.6 相关实体类

/*** 文件搜索VO*/
@Data
public class SearchFileVO {@JsonSerialize(using = ToStringSerializer.class)private Long userFileId;private String fileName;private String filePath;private String extendName;private Long fileSize;private String fileUrl;private Map<String, List<String>> highLight;private Integer isDir;
}
/*** 文件搜索DTO*/
@Data
@JsonIgnoreProperties(ignoreUnknown = true)
public class FileSearchDTO {private String indexName;private String userFileId;private String fileId;private String fileName;private String content;private String fileUrl;private Long fileSize;private Integer storageType;private String identifier;private Long userId;private String filePath;private String extendName;private Integer isDir;private String deleteTime;private String deleteBatchNum;
}
/*** 文件查询条件DTO*/
@Data
public class SearchFileQueryDTO {@ApiModelProperty("文件名")private String fileName;@ApiModelProperty("当前页")private long pageNum;@ApiModelProperty("每页数量")private long pageSize;
}

🚀6 RabbitMQ相关部分

6.1 生产者部分(文件服务)

6.1.1 引入依赖

<!-- nacos -->
<dependency><groupId>com.alibaba.cloud</groupId><artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
</dependency>
<!-- RabbitMQ(我的SpringBoot是2.6.8的) -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

6.1.2 完整application.yml

server:port: 8083
spring:# MySQL配置datasource:driver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://127.0.0.1:3306/(数据库名)?characterEncoding=utf8&serverTimezone=Asia/Shanghai&useSSL=false&allowPublicKeyRetrieval=trueusername: rootpassword: (MySQL密码)# nacos注册的服务名application:name: netdisk-filecloud:nacos:discovery:# 配置注册服务的IP地址server-addr: (IP地址):8848username: nacospassword: nacos# rabbitmq相关的配置rabbitmq:host: (IP地址)port: 5672virtual-host: (虚拟主机名,比如:/file)username: (用户名,默认:guest)password: (密码,默认:guest)

6.1.3 RabbitMQConfig 配置类

@Configuration
public class RabbitMQConfig {// 普通交换机public static final String FILE_EXCHANGE = "file.exchange";// 文件保存相关public static final String QUEUE_FILE_SAVE = "queue.file.save";public static final String KEY_FILE_SAVE = "key.file.save";// 文件删除相关public static final String QUEUE_FILE_REMOVE = "queue.file.remove";public static final String KEY_FILE_REMOVE = "key.file.remove";// 死信相关public static final String DEAD_LETTER_EXCHANGE = "deadLetter.exchange";public static final String DEAD_LETTER_QUEUE = "deadLetter.queue";public static final String KEY_FILE_DEAD_LETTER = "key.file.dead.letter";//延迟队列public static final String DELAY_QUEUE = "delay.queue";/*** 文件保存队列** @return*/@Beanpublic Queue queueFileSave() {return new Queue(QUEUE_FILE_SAVE);}/*** 文件删除队列** @return*/@Beanpublic Queue queueFileRemove() {return new Queue(QUEUE_FILE_REMOVE);}/*** 交换机** @return*/@Beanpublic TopicExchange topicExchange() {return new TopicExchange(FILE_EXCHANGE);}/*** 绑定文件保存队列到交换机** @return*/@Beanpublic Binding bindFileSave() {return BindingBuilder.bind(queueFileSave()).to(topicExchange()).with(KEY_FILE_SAVE);}/*** 绑定文件删除队列到交换机** @return*/@Beanpublic Binding bindFileRemove() {return BindingBuilder.bind(queueFileRemove()).to(topicExchange()).with(KEY_FILE_REMOVE);}/*** 定义延时队列** @return*/@Beanpublic Queue delayQueue() {//设置死信交换机和路由keyreturn QueueBuilder.durable(DELAY_QUEUE)//如果消息过时,则会被投递到当前对应的死信交换机.withArgument("x-dead-letter-exchange", DEAD_LETTER_EXCHANGE)//如果消息过时,死信交换机会根据routing-key投递消息到对应的队列.withArgument("x-dead-letter-routing-key", KEY_FILE_DEAD_LETTER).build();}/*** 定义死信交换机** @return*/@Beanpublic TopicExchange deadLetterExchange() {return new TopicExchange(DEAD_LETTER_EXCHANGE);}/*** 定义死信队列** @return*/@Beanpublic Queue deadLetterQueue() {return new Queue(DEAD_LETTER_QUEUE);}/*** 绑定死信队列到死信交换机** @return*/@Beanpublic Binding deadLetterBinding() {return BindingBuilder.bind(deadLetterQueue()).to(deadLetterExchange()).with(KEY_FILE_DEAD_LETTER);}
}

6.1.4 FileDealComp 文件逻辑处理组件伪代码

/*** 文件逻辑处理组件*/
@Slf4j
@Component
public class FileDealComp {@Autowiredprivate RabbitTemplate rabbitTemplate;private final ThreadPoolExecutor executor = new ThreadPoolExecutor(12, // 核心线程数20, // 最大线程数1, // 线程存活时间TimeUnit.SECONDS, // 存活时间单位new ArrayBlockingQueue<>(1000) // 任务队列);/*** 更新ES数据** @param userFileId*/public void uploadES(Long userFileId) {executor.execute(() -> {FileSearchDTO fileSearchDTO = new FileSearchDTO();// 通过用户文件ID查询用户文件信息...// 通过文件ID查询文件信息...// 将用户文件信息和文件信息同步到fileSearchDTO对象...  // 消息队列更新ESrabbitTemplate.convertAndSend(RabbitMQConfig.FILE_EXCHANGE, RabbitMQConfig.KEY_FILE_SAVE, fileSearchDTO);});}/*** 删除ES数据** @param userFileId*/public void deleteES(Long userFileId) {// 消息队列删除ESrabbitTemplate.convertAndSend(RabbitMQConfig.FILE_EXCHANGE, RabbitMQConfig.KEY_FILE_REMOVE, userFileId);}/*** 分享文件过期处理** @param shareBatchNum 分享批次号*/public void expiredShareFile(String shareBatchNum) {Share share = new Share();// 根据分享批次号获取分享信息...// 将分享信息同步到share对象...// 定义日期格式SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");long differenceInMillis = 0;try {// 解析日期字符串为日期对象Date shareDate = sdf.parse(share.getShareTime());Date endDate = sdf.parse(share.getEndTime());// 计算时间差(毫秒数)differenceInMillis = endDate.getTime() - shareDate.getTime();} catch (ParseException e) {e.printStackTrace();}// 存活时间String expiration = Long.toString(differenceInMillis);// 延时队列rabbitTemplate.convertAndSend(RabbitMQConfig.DELAY_QUEUE, share, message -> {message.getMessageProperties().setExpiration(expiration);return message;});}
}

6.1.5 ExpiredShareFileListener 过期的分享文件处理监听器

@Slf4j
@Component
@RabbitListener(queues = "my-dlx-queue")
public class ExpiredShareFileListener {@Autowiredprivate ShareService shareService;// 死信相关public static final String DEAD_LETTER_EXCHANGE = "deadLetter.exchange";public static final String DEAD_LETTER_QUEUE = "deadLetter.queue";public static final String KEY_FILE_DEAD_LETTER = "key.file.dead.letter";@RabbitListener(bindings = {@QueueBinding(key = KEY_FILE_DEAD_LETTER,value = @Queue(value = DEAD_LETTER_QUEUE, durable = "true"),exchange = @Exchange(value = DEAD_LETTER_EXCHANGE, type = ExchangeTypes.TOPIC, ignoreDeclarationExceptions = "true"))})public void receiveShareMessage(Share share) {log.info("监听到文件过期处理操作:{}", share);// 将share的分享状态改为已过期 → 将share的shareStatus由0改为1...log.info("操作完成:{}", share);}
}

6.2 消费者部分(查询服务)

6.2.1 引入依赖

<!-- RabbitMQ (我的SpringBoot是2.6.8的) -->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-amqp</artifactId>
</dependency>

6.2.2 完整application.yml

server:port: 8084
spring:# MySQL配置datasource:driver-class-name: com.mysql.cj.jdbc.Driverurl: jdbc:mysql://127.0.0.1:3306/(数据库名)?characterEncoding=utf8&serverTimezone=Asia/Shanghai&useSSL=false&allowPublicKeyRetrieval=trueusername: rootpassword: (MySQL密码)# nacos注册的服务名application:name: netdisk-searchcloud:nacos:discovery:# 配置注册服务的IP地址server-addr: (IP地址):8848username: nacospassword: nacosmvc:path match:matching-strategy: ant_path_matcherservlet:multipart:enabled: true# 单个文件最大限制max-file-size: 1024MB# 多个文件最大限制max-request-size: 2048MB# rabbitmq相关的配置rabbitmq:host: (IP地址)port: 5672virtual-host: (虚拟主机名,比如:/file)username: (用户名,默认:guest)password: (密码,默认:guest)# elasticsearch相关的配置
elasticsearch:# ES网关地址hostname: (IP地址)# ES网关端口port: 9200# ES网官方方案scheme: http

6.2.3 FileMQListener 文件处理消息队列监听

@Slf4j
@Component
public class FileMQListener {// 普通交换机public static final String FILE_EXCHANGE = "file.exchange";// 文件保存相关public static final String QUEUE_FILE_SAVE = "queue.file.save";public static final String KEY_FILE_SAVE = "key.file.save";// 文件删除相关public static final String QUEUE_FILE_REMOVE = "queue.file.remove";public static final String KEY_FILE_REMOVE = "key.file.remove";@Autowiredprivate ElasticsearchService elasticsearchService;/*** 监听文件信息添加操作** @param fileSearchDTO*/@RabbitListener(bindings = {@QueueBinding(key = KEY_FILE_SAVE,value = @Queue(value = QUEUE_FILE_SAVE, durable = "true"),exchange = @Exchange(value = FILE_EXCHANGE, type = ExchangeTypes.TOPIC, ignoreDeclarationExceptions = "true"))})public void receiveFileSaveMessage(FileSearchDTO fileSearchDTO) {try {log.info("监听到文件信息添加操作:{}", fileSearchDTO);// 更新ES数据elasticsearchService.uploadES(fileSearchDTO);log.info("添加完成:{}", fileSearchDTO);} catch (Exception ex) {ex.printStackTrace();}}/*** 监听文件信息删除操作** @param userFileId*/@RabbitListener(bindings = {@QueueBinding(key = KEY_FILE_REMOVE,value = @Queue(value = QUEUE_FILE_REMOVE, durable = "true"),exchange = @Exchange(value = FILE_EXCHANGE, type = ExchangeTypes.TOPIC, ignoreDeclarationExceptions = "true"))})public void receiveFileDeleteMessage(Long userFileId) {try {log.info("监听到文件信息删除操作:{}", userFileId);// 删除ES数据elasticsearchService.deleteES(userFileId);log.info("文件信息删除完成:{}", userFileId);} catch (Exception ex) {ex.printStackTrace();}}
}

📫7 代码仓库

netdisk-cloud | Gitee

相关文章:

RabbitMQ实现数据库与ElasticSearch的数据同步和分享文件过期处理

&#x1f388; 1 参考文档 RabbitMQ实现数据库与ElasticSearch的数据同步 | Hannya。-CSDN 企业级开发项目实战——基于RabbitMQ实现数据库、elasticsearch的数据同步 | 波总说先赚它一个小目标-CSDN SPringBoot集成RabbitMQ实现30秒过期删除功能 | 军大君-CSDN &#x1f50d; …...

PyCharm集成开发环境安装、启动与设置

作为非开发工程师职业,大家多多少少都会对编程有抵触,其实没有必要对Python有太大的“戒心" ,把Python当做你的一个工具就可以了。——扎克伯格 一、Python的定义&#xff1a; Python是一个高层次的结合了解释性、编译性、互动性和面向对象的脚本语言。Python的设计具有…...

算法与设计分析--实验一

蛮力算法的设计与分析&#xff08;暴力&#xff09; 这次是某不知名学院开学课程的第一次实验&#xff0c;一共5道题&#xff0c;来自力扣 第一题.216组合总和*力扣题目链接 第一道题是经典的树型回溯 class Solution { public:vector<vector<int>> combinatio…...

ElementUI浅尝辄止28:Dropdown 下拉菜单

将动作或菜单折叠到下拉菜单中。 1.如何使用&#xff1f; 移动到下拉菜单上&#xff0c;展开更多操作。 //通过组件slot来设置下拉触发的元素以及需要通过具名slot为dropdown 来设置下拉菜单。默认情况下&#xff0c;下拉按钮只要hover即可&#xff0c;无需点击也会显示下拉菜…...

jupyter 格式化与快捷键

1、标题&#xff1a; # 一级标题 ## 二级标题 ### 三级标题 2、 加粗文本&#xff1a; **加粗文本** 3、斜体文本&#xff1a; _斜体_ 4、删除线 ~删除线~ 5、高亮文本 高亮文本 6、区块引用 > 我是引用文字 >> 我是第二层 >&g…...

Spring以及SpringBoot/SpringCloud注解

一、SpringBoot/Spring 1、SpringBootApplication 包含Configuration、EnableAutoConfiguration、ComponentScan通常在主类上 其中ComponentScan让Spring Boot扫描到Configuration类并把它加入到程序上下文&#xff0c;如果扫描到有Component Controller Service等这些注解的…...

vim常用操作

一、Esc键 & 命令模式 1.撤销&#xff1a;u 恢复撤销&#xff1a;Ctrl r 2.定位 行首&#xff1a;0 行尾&#xff1a;$ 第7行&#xff1a;7G 3.编辑 下行开始插入&#xff1a; o 删除行&#xff1a;dd 复制3行并粘贴&#xff1a;3yy ---> p 复制单词并粘贴&#…...

Serverless Framework 亚马逊云(AWS)中国地区部署指南

Serverless Framework 亚马逊云(AWS)中国地区部署指南 Serverless Framework 亚马逊云(AWS)中国地区部署指南 前言前置准备 1. 账号的注册2. 全局安装 serverless3. 设置你的系统环境变量4. 设置部署凭证 快速部署一个 hello world 创建入口函数 index.js event 参数context 参…...

【Spring Cloud系统】- 轻量级高可用工具Keepalive详解

【Spring Cloud系统】- 轻量级高可用工具Keepalive详解 文章目录 【Spring Cloud系统】- 轻量级高可用工具Keepalive详解一、概述二、Keepalive分类2.1 TCP的keepalive2.2 HTTP的keep-alive2.3 TCP的 KeepAlive 和 HTTP的 Keep-Alive区别 三、nginx的keepalive配置3.1 nginx保持…...

【JAVA-Day05】深入理解Java数据类型和取值范围

深入理解Java数据类型和取值范围 深入理解Java数据类型和取值范围摘要一、Java的数据类型1.1 存储单位1.2 Java基本数据类型 二、Java的取值范围2.1 变量定义2.2 取值范围验证 三、总结 博主 默语带您 Go to New World. ✍ 个人主页—— 默语 的博客&#x1f466;&#x1f3fb;…...

“JSR303和拦截器在Java Web开发中的应用与实践“

目录 引言JSR303什么是JSR303?为什么要使用JSR303?常用注解快速入门JSR303 拦截器什么是拦截器拦截器与过滤器应用场景快速入门拦截器 总结 引言 在Java Web开发过程中&#xff0c;我们经常会遇到需要对输入数据进行验证和处理&#xff0c;同时需要对请求进行拦截与控制的需…...

第六章 图 六、最小生成树(Prim算法、Kruskal算法)

一、定义 对于一个带权连通无向图G(V,E)&#xff0c;生成树不同&#xff0c;每棵树的权(即树中所有边上的权值之和)也可能不同。设R为G的所有生成树的集合&#xff0c;若T为R中边的权值之和最小的生成树&#xff0c;则T称为G的最小生成树(Minimum-Spanning-Tree, MST)。 二、手…...

机器学习笔记 - 什么是 MLOps?

什么是 MLOps? Machine learning operations (MLOps) 作为一个新兴领域,MLOps 在数据科学家、机器学习工程师和人工智能爱好者中迅速崛起。MLOps 代表机器学习操作。MLOps 是机器学习工程的核心功能,专注于简化将机器学习模型投入生产、然后维护和监控的过程。MLOps 是一种…...

初阶扫雷(超详解)

✨博客主页&#xff1a;小钱编程成长记 &#x1f388;博客专栏&#xff1a;C语言小游戏 &#x1f388;推荐相关博文&#xff1a;初阶三子棋&#xff08;超详解&#xff09; 初阶扫雷 1.游戏介绍2.基本思路3.实现前的准备4.实现步骤4.1 打印菜单4.2 初始化扫雷棋盘4.3 打印扫雷棋…...

计算机视觉CV:1000字总结介绍

目录 1.CV计算机视觉 2.计算机视觉的应用 3.计算机视觉的基本技术 4.计算机视觉的发展趋势 1.CV计算机视觉 计算机视觉&#xff08;Computer Vision, CV&#xff09;是指通过计算机技术模拟人类视觉&#xff0c;让计算机能够“看”懂和理解图像和视频。计算机视觉发展了多…...

JavaScript 之 Symbol 数据类型

一、简介 ​ symbol类型是ES6新引入的一种基本数据类型&#xff0c;该类型具有静态属性和静态方法。其中静态属性暴露了几个内建的成员对象&#xff0c;静态方法暴露了全局的symbol注册。 ​ symbol类型具有以下特点&#xff1a;① 唯一性&#xff1a;每个symbol值都是唯一的…...

在Docker中运行PostgreSQL数据库

1.下载Docker 2.设置DockerHub账号 3.运行Docker并下载Image 4.启动PostgreSQL Image 5.连接到数据库运行SQL docker run --name some-postgres -p 5432:5432 -e POSTGRES_PASSWORDmysecretpassword -d postgres 开放端口从Docker容器到主操作系统&#xff0c;这将允许我们…...

实现Spring Boot集成MyBatis

引言 在Java开发中&#xff0c;Spring Boot和MyBatis是非常常用的框架。Spring Boot是一个快速开发应用程序的框架&#xff0c;而MyBatis是一个持久化框架&#xff0c;可以方便地操作数据库。本文将介绍如何使用Idea集成Spring Boot和MyBatis&#xff0c;并创建一个简单的示例…...

关于算法的时间复杂度(度量算法执行时间的两种方法、渐进时间复杂度、时间复杂度的几个性质、渐进估算、常见的渐进时间复杂度排序)

目录 度量算法执行时间的两种方法 事后统计法&#xff08;Post Hoc Analysis&#xff09;&#xff1a; 事前统计法&#xff08;Pre Hoc Analysis&#xff09;&#xff1a; 渐进时间复杂度 时间复杂度的几个性质 渐进估算 常见的渐进时间复杂度排序 度量算法执行时间的两…...

SpringBoot项目--电脑商城【显示商品详情功能】

1.持久层[Mapper] 1规划需要执行的SQL语句 根据商品id显示商品详情的SQL语句 SELECT * FROM t_product WHERE id?2 设计接口和抽象方法 在ProductMapper接口中添加抽象方法 /*** 根据商品id查询商品详情* param id 商品id* return 匹配的商品详情&#xff0c;如果没有匹配…...

synchronized 学习

学习源&#xff1a; https://www.bilibili.com/video/BV1aJ411V763?spm_id_from333.788.videopod.episodes&vd_source32e1c41a9370911ab06d12fbc36c4ebc 1.应用场景 不超卖&#xff0c;也要考虑性能问题&#xff08;场景&#xff09; 2.常见面试问题&#xff1a; sync出…...

AI书签管理工具开发全记录(十九):嵌入资源处理

1.前言 &#x1f4dd; 在上一篇文章中&#xff0c;我们完成了书签的导入导出功能。本篇文章我们研究如何处理嵌入资源&#xff0c;方便后续将资源打包到一个可执行文件中。 2.embed介绍 &#x1f3af; Go 1.16 引入了革命性的 embed 包&#xff0c;彻底改变了静态资源管理的…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...

QT3D学习笔记——圆台、圆锥

类名作用Qt3DWindow3D渲染窗口容器QEntity场景中的实体&#xff08;对象或容器&#xff09;QCamera控制观察视角QPointLight点光源QConeMesh圆锥几何网格QTransform控制实体的位置/旋转/缩放QPhongMaterialPhong光照材质&#xff08;定义颜色、反光等&#xff09;QFirstPersonC…...

Java数值运算常见陷阱与规避方法

整数除法中的舍入问题 问题现象 当开发者预期进行浮点除法却误用整数除法时,会出现小数部分被截断的情况。典型错误模式如下: void process(int value) {double half = value / 2; // 整数除法导致截断// 使用half变量 }此时...

Proxmox Mail Gateway安装指南:从零开始配置高效邮件过滤系统

&#x1f49d;&#x1f49d;&#x1f49d;欢迎莅临我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐&#xff1a;「storms…...

从“安全密码”到测试体系:Gitee Test 赋能关键领域软件质量保障

关键领域软件测试的"安全密码"&#xff1a;Gitee Test如何破解行业痛点 在数字化浪潮席卷全球的今天&#xff0c;软件系统已成为国家关键领域的"神经中枢"。从国防军工到能源电力&#xff0c;从金融交易到交通管控&#xff0c;这些关乎国计民生的关键领域…...

ubuntu22.04有线网络无法连接,图标也没了

今天突然无法有线网络无法连接任何设备&#xff0c;并且图标都没了 错误案例 往上一顿搜索&#xff0c;试了很多博客都不行&#xff0c;比如 Ubuntu22.04右上角网络图标消失 最后解决的办法 下载网卡驱动&#xff0c;重新安装 操作步骤 查看自己网卡的型号 lspci | gre…...

绕过 Xcode?使用 Appuploader和主流工具实现 iOS 上架自动化

iOS 应用的发布流程一直是开发链路中最“苹果味”的环节&#xff1a;强依赖 Xcode、必须使用 macOS、各种证书和描述文件配置……对很多跨平台开发者来说&#xff0c;这一套流程并不友好。 特别是当你的项目主要在 Windows 或 Linux 下开发&#xff08;例如 Flutter、React Na…...

前端高频面试题2:浏览器/计算机网络

本专栏相关链接 前端高频面试题1&#xff1a;HTML/CSS 前端高频面试题2&#xff1a;浏览器/计算机网络 前端高频面试题3&#xff1a;JavaScript 1.什么是强缓存、协商缓存&#xff1f; 强缓存&#xff1a; 当浏览器请求资源时&#xff0c;首先检查本地缓存是否命中。如果命…...