TurtleBot3 Package turtlebot3_drive source code read
前言
此处阅读简单的 turtlebot3_drive 代码。
从ROS的角度,作为一个demo,它足够小、简单,可以从中看见ROS的 NodeHandle如何使用。此外,我们也能简单地看到 “自动避障功能的实现”。
从C++的角度,它实际上并不复杂,我们能够回顾C++编码的规范,对于初学者有帮助。
turtlebot3_drive.h
/*******************************************************************************
* Copyright 2016 ROBOTIS CO., LTD.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*******************************************************************************//* Authors: Taehun Lim (Darby) */#ifndef TURTLEBOT3_DRIVE_H_
#define TURTLEBOT3_DRIVE_H_#include <ros/ros.h>#include <sensor_msgs/LaserScan.h>
#include <geometry_msgs/Twist.h>
#include <nav_msgs/Odometry.h>#define DEG2RAD (M_PI / 180.0)
#define RAD2DEG (180.0 / M_PI)#define CENTER 0
#define LEFT 1
#define RIGHT 2#define LINEAR_VELOCITY 0.3
#define ANGULAR_VELOCITY 1.5#define GET_TB3_DIRECTION 0
#define TB3_DRIVE_FORWARD 1
#define TB3_RIGHT_TURN 2
#define TB3_LEFT_TURN 3class Turtlebot3Drive
{public:Turtlebot3Drive();~Turtlebot3Drive();bool init();bool controlLoop();private:// ROS NodeHandleros::NodeHandle nh_;ros::NodeHandle nh_priv_; // not see your funciton here, maybe for further extension// ROS Parameters// ROS Time// ROS Topic Publishersros::Publisher cmd_vel_pub_;// ROS Topic Subscribersros::Subscriber laser_scan_sub_;ros::Subscriber odom_sub_;// Variablesdouble escape_range_;double check_forward_dist_;double check_side_dist_;double scan_data_[3] = {0.0, 0.0, 0.0};double tb3_pose_;double prev_tb3_pose_; // 2 variables to keep track// Function prototypesvoid updatecommandVelocity(double linear, double angular);void laserScanMsgCallBack(const sensor_msgs::LaserScan::ConstPtr &msg);void odomMsgCallBack(const nav_msgs::Odometry::ConstPtr &msg);
};
#endif // TURTLEBOT3_DRIVE_H_
一般而言,我们先看 .h 文件,了解接口。在文件的最上方,是协议。此外,我们看到了头文件的包含,#define 定义的宏。这些宏通常作为常量,方便编码,提高可读性。
在class内部,有两个接口,init() 和 controlLoop()。其中 init()负责初始化工作,controlLoop()是控制循环,于是我们知道,实际的实现文件中,可能存在一个循环,其中起作用的就是controlLoop()方法。
在 private部分,定义了变量,有注释进行说明。最后,还有三个 helper function/ utility function,用于辅助public函数功能的实现,但对外隐蔽,提高了封装性。此外有两个变量记录着 turtle的姿态。
turtlebot3_drive.cpp
/*******************************************************************************
* Copyright 2016 ROBOTIS CO., LTD.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*******************************************************************************//* Authors: Taehun Lim (Darby) */#include "turtlebot3_gazebo/turtlebot3_drive.h"Turtlebot3Drive::Turtlebot3Drive(): nh_priv_("~")
{//Init gazebo ros turtlebot3 nodeROS_INFO("TurtleBot3 Simulation Node Init");auto ret = init();ROS_ASSERT(ret);
}Turtlebot3Drive::~Turtlebot3Drive()
{updatecommandVelocity(0.0, 0.0);ros::shutdown();
}/*******************************************************************************
* Init function
*******************************************************************************/
bool Turtlebot3Drive::init()
{// initialize ROS parameterstd::string cmd_vel_topic_name = nh_.param<std::string>("cmd_vel_topic_name", "");// initialize variablesescape_range_ = 30.0 * DEG2RAD;check_forward_dist_ = 0.7;check_side_dist_ = 0.6;tb3_pose_ = 0.0; // so there is only 2 kind of pose recorded prev_tb3_pose_ = 0.0; // record the previous pose // initialize publisherscmd_vel_pub_ = nh_.advertise<geometry_msgs::Twist>(cmd_vel_topic_name, 10);// initialize subscriberslaser_scan_sub_ = nh_.subscribe("scan", 10, &Turtlebot3Drive::laserScanMsgCallBack, this); // asynchronous, callback funcitonodom_sub_ = nh_.subscribe("odom", 10, &Turtlebot3Drive::odomMsgCallBack, this); // the topic name i have seenreturn true;
}void Turtlebot3Drive::odomMsgCallBack(const nav_msgs::Odometry::ConstPtr &msg)
{double siny = 2.0 * (msg->pose.pose.orientation.w * msg->pose.pose.orientation.z + msg->pose.pose.orientation.x * msg->pose.pose.orientation.y);double cosy = 1.0 - 2.0 * (msg->pose.pose.orientation.y * msg->pose.pose.orientation.y + msg->pose.pose.orientation.z * msg->pose.pose.orientation.z); tb3_pose_ = atan2(siny, cosy);
}void Turtlebot3Drive::laserScanMsgCallBack(const sensor_msgs::LaserScan::ConstPtr &msg)
{uint16_t scan_angle[3] = {0, 30, 330};for (int num = 0; num < 3; num++){if (std::isinf(msg->ranges.at(scan_angle[num]))){scan_data_[num] = msg->range_max;}else{scan_data_[num] = msg->ranges.at(scan_angle[num]);}}
}void Turtlebot3Drive::updatecommandVelocity(double linear, double angular)
{geometry_msgs::Twist cmd_vel;cmd_vel.linear.x = linear;cmd_vel.angular.z = angular;cmd_vel_pub_.publish(cmd_vel);
}/*******************************************************************************
* Control Loop function
*******************************************************************************/
bool Turtlebot3Drive::controlLoop()
{static uint8_t turtlebot3_state_num = 0;switch(turtlebot3_state_num){case GET_TB3_DIRECTION:if (scan_data_[CENTER] > check_forward_dist_){if (scan_data_[LEFT] < check_side_dist_){prev_tb3_pose_ = tb3_pose_;turtlebot3_state_num = TB3_RIGHT_TURN;}else if (scan_data_[RIGHT] < check_side_dist_){prev_tb3_pose_ = tb3_pose_;turtlebot3_state_num = TB3_LEFT_TURN;}else{turtlebot3_state_num = TB3_DRIVE_FORWARD;}}if (scan_data_[CENTER] < check_forward_dist_){prev_tb3_pose_ = tb3_pose_;turtlebot3_state_num = TB3_RIGHT_TURN; // default, turn right when there is an obstacle}break;case TB3_DRIVE_FORWARD: // your logic is the same, after update the velocity, switch to get info thenupdatecommandVelocity(LINEAR_VELOCITY, 0.0);turtlebot3_state_num = GET_TB3_DIRECTION;break;case TB3_RIGHT_TURN:if (fabs(prev_tb3_pose_ - tb3_pose_) >= escape_range_)turtlebot3_state_num = GET_TB3_DIRECTION;elseupdatecommandVelocity(0.0, -1 * ANGULAR_VELOCITY);break;case TB3_LEFT_TURN:if (fabs(prev_tb3_pose_ - tb3_pose_) >= escape_range_)turtlebot3_state_num = GET_TB3_DIRECTION;elseupdatecommandVelocity(0.0, ANGULAR_VELOCITY);break;default:turtlebot3_state_num = GET_TB3_DIRECTION;break;}return true;
}/*******************************************************************************
* Main function
*******************************************************************************/
int main(int argc, char* argv[])
{ros::init(argc, argv, "turtlebot3_drive");Turtlebot3Drive turtlebot3_drive;ros::Rate loop_rate(125);while (ros::ok()){turtlebot3_drive.controlLoop();ros::spinOnce();loop_rate.sleep();}return 0;
}
cstr 输出一份 ROS log,将具体的工作交由 init()方法实现,并由assert()判断实现是否成功初始化。
init()内部对几个 private member进行了初始化,并通过 Nodehandle给 ROS的sub、pub变量声明了相关的 topic,queue size。对于 sub,它还声明了回调函数,注意到 sub 的后两个参数,一个提供了类方法的地址,另一个是 指针 this,指明该类自身。这是C++回调函数的典型写法,原因显而易见:需要给出哪个类的哪个方法,最后的 this 指明该类的 “此对象”。
controlLoop()实际上是一个 switch语句,其中有各种case标识,无非是标识几种状态,在对应的状态中,对laser读得的数据进行判断,并执行各种操作,实现自主前行,自动避障。注意到,雷达存在扫描范围,所以在方法 laserScanMsgCallBack(), 我们对接收到的距离做了判断,并根据情况赋予不同的值。
注意成员函数odomMsgCallBack() 会改变 turtle的姿态,会影响controlLoop()的条件判断和具体语句执行。updatecommandVelocity则改变速度。
attention
这里存在几个细节:
1、controlloop()中,初始化了一个 static 变量。它的生存期从该函数开始执行到节点结束,作用域在函数内部。static还保证了第二次调用该函数时,它不会反复初始化,符合我们的期待。放入函数内部表明,它实际只跟函数相关。
2、main()函数中,使用了一个 while循环,设定了 ros::rate(),定期唤醒当前节点,一种典型的 ros sub端的处理方式。
3、对语句
std::isinf(msg->ranges.at(scan_angle[num]))
std::isinf() 是 STL的部分,msg为ROS的一种数据格式。ranges实际为被包括的 std::vector<float>,详见 ROS Message Type, at()是 std::vector<>的一种方法,比 [] 要安全,如果索引超出范围,它抛出异常。
Summerize
上述的代码很简单,没有复杂的逻辑。可以从订阅topic和发布topic得知,它无法单独完成工作,必须和其他节点配合。
相关文章:
TurtleBot3 Package turtlebot3_drive source code read
前言 此处阅读简单的 turtlebot3_drive 代码。 从ROS的角度,作为一个demo,它足够小、简单,可以从中看见ROS的 NodeHandle如何使用。此外,我们也能简单地看到 “自动避障功能的实现”。 从C的角度,它实际上并不复杂&…...
机器学习的下一个前沿是因果关系吗?
如今,越来越多研究人员意识到,将因果关系融入机器学习,或许会是该领域实现重大突破的关键所在。 机器学习凭借先进的预测能力,已为诸多行业带来了显著变革,但也暴露出了一定的局限性。而因果关系,作为理解…...
Mujoco xml模型
Mujoco xml模型 一个例子compileroptionassetmesh default基本使用childclass与class多个class worldbodybody关系inertialjointgeom XML主要分为以下三个部分: < asset> : 用 tag导入STL文件;< worldbody>:用tag定义…...
MyBatis 详解及代码示例
MyBatis 是一个 半自动 ORM 框架,主要用于 Java 与数据库之间的持久化操作,它本质是对 JDBC 的封装 全名:MyBatis(前身 iBATIS)核心作用:自动将 SQL 执行结果映射为 Java 对象;也可以将 Java 对…...
STL-list链表
STL-list链表实现 STL中采用双向带头循环链表来实现 list,下面将使用 C++ 实现 STL list 链表。 list 类中包含两个主要部分,一个是指向哨兵位头节点的指针(_head),另一个是结构体类型的迭代器(__list_iterator)。 哨兵位头节点本身是不存储数据的,它只是用于简化代码…...
R语言中的rvest库写个视频爬虫通用代码
朋友让我用R语言的rvest库写一个通用的视频爬虫代码示例。首先,我需要回忆一下rvest库的主要功能,它主要是用来做网页抓取和解析的,类似于Python的BeautifulSoup。但是视频爬虫的话,可能需要处理动态加载的内容,或者找…...
SQLite 中日期型数据定义及处理(Delphi 版本)
在使用SQLite的时候,肯定需要使用到日期型数据类型,但是SQLite没有直接支持日期类型,比如在其他数据库中支持的DateTime类型,在Delphi中是TDateTime类型。 那么实际处理中应该如何处理呢? 可以使用两种方式类在SQLit…...
4.9复习记
1.地宫取宝--记忆化搜索,可以先写void dfs,然后在改成ll 形式的,边界条件return 0/1; 记忆化数组与dfs元素保持一致,记得记忆化剪枝 这个题特殊在value可能是0,不取的时候应该记为-1 https://mpbeta.cs…...
Flink基础
Flink基础 目录 Flink简介核心概念编程模型核心功能应用场景部署模式生态系统最佳实践学习资源实践案例高级特性 1. Flink简介 1.1 什么是Flink Apache Flink是一个开源的分布式流处理和批处理系统。它能够处理有界(批处理)和无界(流处理…...
SASE、零信任安全理念的发展脉络
SASE(安全访问服务边缘)与零信任架构的发展脉络,是云安全理念从 “边界防御” 向 “动态信任” 跃迁的典型缩影。两者的演进既独立又交织,共同推动网络安全从静态合规走向主动治理。以下从技术起源、理念突破、产业实践到未来趋势展开深度解析: 一、零信任:从理论构想到…...
CompletableFuture 和 List<CompletableFuture> allOf() join() get() 使用经验
CompletableFuture<Map<Menu, Map<IntentDetail, Double>>> xxx CompletableFuture.supplyAsync(() -> {Map<Menu, Map<IntentDetail, Double>> scores new ConcurrentHashMap<>();// 存储结果scores.computeIfAbsent(menu, k -> n…...
Vue.js组件化开发实战:从工程化到安全纵深设计
文章目录 开篇:现代前端组件化演进之路 组件设计核心:高内聚低耦合实践 工程化基石:从Webpack到Monorepo 安全纵深设计:RASP在组件层的实现 实战:动态表单组件的三次进化 进阶篇:组件工厂模式与策略模…...
【深度解析】SkyWalking 10.2.0版本安全优化与性能提升实战指南
前言 Apache SkyWalking 作为云原生可观测性领域的佼佼者,在微服务架构监控中扮演着至关重要的角色。然而,官方版本在安全性、镜像体积和功能扩展方面仍有优化空间。本文将分享一套完整的 SkyWalking 10.2.0 版本优化方案,从安全漏洞修复到镜…...
NOIP2011提高组.玛雅游戏
目录 题目算法标签: 模拟, 搜索, d f s dfs dfs, 剪枝优化思路*详细注释版代码精简注释版代码 题目 185. 玛雅游戏 算法标签: 模拟, 搜索, d f s dfs dfs, 剪枝优化 思路 可行性剪枝 如果某个颜色的格子数量少于 3 3 3一定无解因为要求字典序最小, 因此当一个格子左边有…...
常微分方程求解全解析:从基础到矩阵方法深度实践
常微分方程求解全解析:从基础到矩阵方法深度实践 一、常微分方程基础与解法体系 1.微分方程基本概念解析 常微分方程的阶数指方程中未知函数导数的最高阶数。通解是包含任意常数且常数个数与方程阶数相同的解,特解则是通解中任意常数取特定值得到的解。以自由落体运动为例…...
Go 微服务框架 | 中间件
文章目录 定义中间件前置中间件后置中间件路由级别中间件 定义中间件 中间件的作用是给应用添加一些额外的功能,但是不会影响原有应用的编码方式,想用的时候直接添加,不想用的时候也可以轻松去除,实现所谓的可插拔。中间件的实现…...
【HarmonyOS Next之旅】DevEco Studio使用指南(十二)
目录 1 -> Code Linter代码检查 2 -> 配置代码检查规则 3 -> 查看/处理代码检查结果 1 -> Code Linter代码检查 Code Linter针对ArkTS/TS代码进行最佳实践/编程规范方面的检查。 可根据扫描结果中告警提示手工修复代码缺陷,或者执行一键式自动修复…...
Java设计模式之桥接模式:从入门到架构级实践
1. 什么是桥接模式? 桥接模式(Bridge Pattern) 是一种结构型设计模式,其核心目标是将抽象部分与实现部分分离,使它们能够独立变化。通过这种方式,桥接模式解决了多层继承带来的复杂性,并增强了…...
Jupyter Lab 无法启动 Kernel 问题排查与解决总结
📄 Jupyter Lab 无法启动 Kernel 问题排查与解决总结 一、问题概述 🚨 现象描述: 用户通过浏览器访问远程服务器的 Jupyter Lab 页面(http://xx.xx.xx.xx:8891/lab)后,.ipynb 文件可以打开,但无…...
【LeetCode 热题100】73:矩阵置零(详细解析)(Go语言版)
🚀 力扣热题 73:矩阵置零(详解 多种解法) 📌 题目描述 给定一个 m x n 的整数矩阵 matrix,如果一个元素为 0,则将其所在行和列的所有元素都设为 0。请你 原地 使用常量空间解决。 Ἲ…...
OminiAdapt:学习跨任务不变性,实现稳健且环境-觉察的机器人操作
25年3月来自中科大、北理工和中科院自动化所的论文“OminiAdapt: Learning Cross-Task Invariance for Robust and Environment-Aware Robotic Manipulation”。 随着具身智能的快速发展,利用大规模人体数据对人形机器人进行高水平的模仿学习,成为学术界…...
Vue3中父组件将一个ref定义的对象类型传递给子组件的解包机制
在Vue3中,当父组件将一个ref定义的对象类型传递给子组件时,子组件接收到的不是原始的Ref类型,而是该ref的.value值,即被解包后的响应式对象。具体行为如下: 关键点解析: 自动解包机制: Vue3在模…...
批量将 SVG 转换为 jpg/png/Word/PDF/ppt 等其它格式
SVG(可缩放矢量图形)是一种广泛使用的图像格式,因其矢量特性在不同分辨率下都能保持清晰,但在某些情况下,用户可能需要将 SVG 格式的图片转换为更常见的位图格式,如 JPG、PNG 等,以适应不同平台…...
微服务篇——SpringCloud
服务注册 Spring Cloud5大组件有哪些? 服务注册和发现是什么意思?Spring Cloud如何实现服务注册发现? nacos与eureka的区别 负载均衡 如何实现负载均衡? Ribbon负载均衡的策略有哪些? 如何自定义负载均衡的策略&…...
Windows 11 家庭中文版 安装docker desktop 无法开启自启动问题处理
前言 我在某台Windows 11家庭中文版的电脑上安装Docker Desktop后,老是无法开机启动,已经按照Docker Desktop 设置调整的方式设置了开机启动,但是重启后发现还是无法自启动,需要手动点击启动。然后使用任务计划程序新建一个开机启…...
蓝桥杯备考
先浅学一遍数据结构,不会拉倒,找点简单题练练语法基础 然后边学边刷二分查找和双指针 递归和暴力,边学边刷 学习贪心,练个几十道 再去过下数据结构 开始算法:搜索,动态规划, 搜索很重要,深…...
Elasticsearch 系列专题 - 第一篇:Elasticsearch 入门
Elasticsearch 是一个功能强大的开源分布式搜索和分析引擎,广泛应用于日志分析、实时搜索、数据可视化等领域。本篇将带你了解 Elasticsearch 的基本概念、安装方法以及简单操作,帮助你快速上手。 1. 什么是 Elasticsearch? 1.1 Elasticsearch 的定义与核心概念 Elasticse…...
【LeetCode 题解】数据库:1321.餐馆营业额变化增长
一、问题描述 本题给定了一个名为 Customer 的表,记录了餐馆顾客的交易数据,包括顾客 ID、姓名、访问日期和消费金额。作为餐馆老板,我们的任务是分析营业额的变化增长情况,具体来说,就是计算以 7 天(某日…...
Apache Nifi安装与尝试
Apache NIFI中文文档 地址:https://nifichina.github.io/ 下载安装配置 1、环境准备 Nifi的运行需要依赖于java环境,所以本机上需要安装java环境,并配置环境变量。 1.1查看本机是否已经存在java环境 请先执行以下命令找出系统中真实可用…...
【Git 常用操作指令指南】
一、初始化与配置 1. 设置全局账户信息 git config --global user.name "用户名" # 设置全局用户名 git config --global user.email "邮箱" # 设置全局邮箱 --global 表示全局生效,若需针对单个仓库配置,可省略该参数 2.…...
Django 生成PDF文件
在这里,我们将学习如何使用Django视图设计和生成PDF文件。我们将使用ReportLab Python PDF库生成PDF,该库可以创建定制的动态PDF文件。 这是一个开源库,可以通过在Ubuntu中使用以下命令轻松下载。 $ pip install reportlab Python Copy …...
多账户使用Github的场景,设置 SSH 多账号使用特定 key
遇到多账户使用Github的场景,常难以管理ssh文件 解决方案: 你可以通过配置 ~/.ssh/config 文件,生成多个SSH key 让 Git 识别不同 key 来对应不同 GitHub 账号。 ✅ 正确的 key 类型有这些常见选项: rsa:老牌算法&a…...
js中this指向问题
在js中,this关键字的指向是一个比较重要的概念,它的值取决于函数的调用方式。 全局状态下 //全局状态下 this指向windowsconsole.log("this", this);console.log("thiswindows", this window); 在函数中 // 在函数中 this指向win…...
BabelDOC ,开源的 AI PDF 翻译工具
BabelDOC 是一款开源智能 PDF 翻译工具,专门为科学论文的翻译而设计。它能够在原文旁边生成翻译文本,实现双语对照,用户无需频繁切换窗口,极大提升了阅读的便利性。此外,BabelDOC 能够完整保留数学公式、表格和图形&am…...
Dify 生成提示词的 Prompt
Dify 生成提示词的 Prompt **第1次提示词****第2次提示词****第3次提示词**总结 Dify 生成提示词是,会和LLM进行3次交互,下面是和LLM进行交互是的Prompt。 以下是每次提示词的概要、目标总结以及原始Prompt: 第1次提示词 概要: …...
在nvim的snippet补全片段中增加函数注释的功能
一、补全片段路径 如果使用nvim,应当在nvim的snippet的插件中增加对应补全的片段,目前我所用的补全的片段路径如下: /home/zhaoky/.local/share/nvim/site/pack/packer/start/vim-snippets.git/snippets我当前补全的是c语言所以使用的片段是c.snippets…...
阿里云负载均衡为何费用高昂?——深度解析技术架构与市场定价策略
本文深度解析阿里云负载均衡(SLB)产品的定价体系,从技术架构、安全防护、合规成本三个维度揭示费用构成逻辑。通过2023年某跨国企业遭受的混合型DDoS攻击案例,结合Gartner最新安全支出报告,给出企业级负载均衡成本优化…...
大数据(7)Kafka核心原理揭秘:从入门到企业级实战应用
目录 一、大数据时代的技术革命1.1 消息中间件演进史1.2 Kafka核心设计哲学 二、架构深度解构2.1 核心组件拓扑2.1.1 副本同步机制(ISR) 2.2 生产者黑科技2.3 消费者演进路线 三、企业级应用实战3.1 金融行业实时风控3.2 物联网数据管道 四、生产环境优化…...
01背包 Java
① 记忆化搜索解法: import java.util.*; import java.io.*;public class Main {static int n, m;static int[] v, w;static int[][] memory; // 记忆化数组public static void main(String[] args) throws Exception {BufferedReader br new BufferedReader(new …...
【Kafka基础】消费者命令行完全指南:从基础到高级消费
Kafka消费者是消息系统的关键组成部分,掌握/export/home/kafka_zk/kafka_2.13-2.7.1/bin/kafka-console-consumer.sh工具的使用对于调试、测试和监控都至关重要。本文将全面介绍该工具的各种用法,帮助您高效地从Kafka消费消息。 1 基础消费模式 1.1 从最…...
Seq2Seq - 编码器(Encoder)和解码器(Decoder)
本节实现一个简单的 Seq2Seq(Sequence to Sequence)模型 的编码器(Encoder)和解码器(Decoder)部分。 重点把握Seq2Seq 模型的整体工作流程 理解编码器(Encoder)和解码器(…...
@SchedulerLock 防止分布式环境下定时任务并发执行
背景 在一个有多个服务实例的分布式系统中,如果你用 Scheduled 来定义定时任务,所有实例都会执行这个任务。ShedLock 的目标是只让一个实例在某一时刻执行这个定时任务。 使用步骤 引入依赖 当前以redisTemplate为例子,MongoDB、Zookeeper…...
【力扣hot100题】(077)跳跃游戏
我最开始的想法还是太单纯了,最开始想着用回溯法,然后想到上一题的经验又想到了动态规划,虽然知道贪心题不太可能会这么复杂但实在想不出别的办法……果然我的智商做贪心题的极限就只能达到找零问题那种水平…… 最开始的方法,击…...
多光谱相机:林业监测应用(病虫害、外来物种、森林防火识别)
随着气候变暖和人类活动的增加,森林火灾发生的频率和强度都有所上升,而我国森林防火基础设施薄弱,监测预警体系不够完善,扑救能力和应急响应能力有待提高。气候变化导致气温升高、降水分布不均等,影响了树木的生长和发…...
Dynamic Programming(LeetCode 740)
740. 删除并获得点数 相关企业提示给你一个整数数组 nums ,你可以对它进行一些操作。 每次操作中,选择任意一个 nums[i] ,删除它并获得 nums[i] 的点数。之后,你必须删除 所有 等于 nums[i] - 1 和 nums[i] 1 的元素。 开始你…...
虚拟列表react-virtualized使用(npm install react-virtualized)
1. 虚拟化列表 (List) // 1. 虚拟化列表 (List)import { List } from react-virtualized; import react-virtualized/styles.css; // 只导入一次样式// 示例数据 const list Array(1000).fill().map((_, index) > ({id: index,name: Item ${index},description: This is i…...
[特殊字符] 手机连接车机热点并使用 `iperf3` 测试网络性能
好的,以下是根据你的描述整理出来的步骤及解释: 📶 手机连接车机热点并使用 iperf3 测试网络性能 本文将通过 iperf3 来测试手机和车机之间的网络连接性能。我们会让车机作为服务端,手机作为客户端,进行 UDP 流量传输…...
C#,VB.NET正则表达式法替换代码
如何设置必须是MGBOX开头, msgbox这种注释自动跳过 在 Visual Studio 中使用 Ctrl H 进行替换操作时,若要确保仅替换以 MsgBox 开头的代码,同时跳过注释里的 MsgBox,可以利用正则表达式来实现。以下为你详细介绍操作步骤: 1. 打…...
从MySQL快速上手大数据Hive
从MySQL快速上手大数据Hive Hive简介 hive是基于Hadoop构建的一套数据仓库分析系统,它提供了丰富的SQL查询方式(DML)来分析存储在Hadoop分布式文件系统中的数据: 可以将结构化的数据文件映射为一张数据库表,并提供完整的SQL查…...
基于华为云kubernetes的应用多活的示例
1 概述 为避免地域级别的故障,需要将单机房架构变成双地域架构(两个机房物理距离越远,网络延时越大,网延时是业务研发首先关注的)。单边写的多机房架构,是落地性比较大的一个方案,相对于单元化…...