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

基于Piecewise Jerk Speed Optimizer的速度规划算法(附ROS C++/Python仿真)

目录

  • 1 时空解耦运动规划
  • 2 PJSO速度规划原理
    • 2.1 优化变量
    • 2.2 代价函数
    • 2.3 约束条件
    • 2.4 二次规划形式
  • 3 算法仿真
    • 3.1 ROS C++仿真
    • 3.2 Python仿真

1 时空解耦运动规划

在自主移动系统的运动规划体系中,时空解耦的递进式架构因其高效性与工程可实现性被广泛采用。这一架构将复杂的高维运动规划问题分解为路径规划与速度规划两个层次化阶段:路径规划阶段基于静态环境约束生成无碰撞的几何轨迹;速度规划阶段则在此基础上引入时间维度,通过动态障碍物预测、运动学约束建模等为机器人或车辆赋予时间轴上的运动规律。这种解耦策略虽在理论上牺牲了时空联合优化的最优性,却通过分层降维大幅提升了复杂场景下的计算效率,使其成为自动驾驶、服务机器人等实时系统的经典方案。

在这里插入图片描述

2 PJSO速度规划原理

2.1 优化变量

分段加加速度优化(Piecewise Jerk Speed Optimizer, PJSO)算法是常用的纵向速度优化方式,核心原理是用三次多项式表示s-t图中每个时间区间 [ t k , t k + 1 ) \left[ t_k,t_{k+1} \right) [tk,tk+1)的速度曲线,并约束每个区间加加速度相等,其中 k = 0 , 1 , ⋯ , n − 2 k=0,1,\cdots ,n-2 k=0,1,,n2, 可取为路径规划的路点数量。PJSO算法的优化变量为

x = [ l s 0 s ˙ 0 s ¨ 0 ⋯ s k s ˙ k s ¨ k ⋯ s n − 1 s ˙ n − 1 s ¨ n − 1 ] 3 n × 1 T \boldsymbol{x}=\left[ \begin{matrix}{l} s_0& \dot{s}_0& \ddot{s}_0& \cdots& s_k& \dot{s}_k& \ddot{s}_k& \cdots& s_{n-1}& \dot{s}_{n-1}& \ddot{s}_{n-1}\\\end{matrix} \right] _{3n\times 1}^{T} x=[ls0s˙0s¨0sks˙ks¨ksn1s˙n1s¨n1]3n×1T

2.2 代价函数

代价函数可设计为

J = ∑ i = 0 n − 2 ( w s ( s i − s i , r e f ) 2 + w v ( s ˙ i − s ˙ i , r e f ) 2 + w a s ¨ i 2 + w j s i ( 3 ) 2 ) + w s , e n d ( s n − 1 − s e n d ) 2 + w v , e n d ( s ˙ n − 1 − s ˙ e n d ) 2 + w a , e n d ( s ¨ n − s ¨ e n d ) 2 J=\sum_{i=0}^{n-2}{\left( w_s\left( s_i-s_{i,\mathrm{ref}} \right) ^2+w_v\left( \dot{s}_i-\dot{s}_{i,\mathrm{ref}} \right) ^2+w_a\ddot{s}_{i}^{2}+w_j{s}_{i}^{(3)2} \right)}\\+w_{s,\mathrm{end}}\left( s_{n-1}-s_{\mathrm{end}} \right) ^2+w_{v,\mathrm{end}}\left( \dot{s}_{n-1}-\dot{s}_{\mathrm{end}} \right) ^2+w_{a,\mathrm{end}}\left( \ddot{s}_n-\ddot{s}_{\mathrm{end}} \right) ^2 J=i=0n2(ws(sisi,ref)2+wv(s˙is˙i,ref)2+was¨i2+wjsi(3)2)+ws,end(sn1send)2+wv,end(s˙n1s˙end)2+wa,end(s¨ns¨end)2

通过

s i ( 3 ) = s ¨ i + 1 − s ¨ i Δ t {s}_i^{(3)}=\frac{\ddot{s}_{i+1}-\ddot{s}_i}{\Delta t} si(3)=Δts¨i+1s¨i

隐式地消除加加速度变量,代入代价函数并消除常数项可化简为

J = w s ∑ i = 0 n − 2 s i 2 + w s , e n d s n − 1 2 − 2 w s ∑ i = 0 n − 2 s i , r e f s i − 2 w s , e n d s e n d s n − 1 + w v ∑ i = 0 n − 2 s ˙ i 2 + w v , e n d s ˙ n − 1 2 − 2 w v ∑ i = 0 n − 2 s ˙ i , r e f s ˙ i − 2 w v , e n d s ˙ e n d s ˙ n − 1 + ( w a + w j Δ t 2 ) s ¨ 0 2 + ( w a + w a , e n d + w j Δ t 2 ) s ¨ n − 1 2 − 2 w a , e n d s ¨ e n d s ¨ n − 1 + ( w a + 2 w j Δ t 2 ) ∑ i = 1 n − 2 s ¨ i 2 − 2 w j Δ t 2 ∑ i = 0 n − 2 s ¨ i s ¨ i + 1 \begin{aligned}J=&w_s\sum_{i=0}^{n-2}{s_{i}^{2}}+w_{s,\mathrm{end}}s_{n-1}^{2}-2w_s\sum_{i=0}^{n-2}{s_{i,\mathrm{ref}}s_i}-2w_{s,\mathrm{end}}s_{\mathrm{end}}s_{n-1}\\&+w_v\sum_{i=0}^{n-2}{\dot{s}_{i}^{2}}+w_{v,\mathrm{end}}\dot{s}_{n-1}^{2}-2w_v\sum_{i=0}^{n-2}{\dot{s}_{i,\mathrm{ref}}\dot{s}_i}-2w_{v,\mathrm{end}}\dot{s}_{\mathrm{end}}\dot{s}_{n-1}\\&+\left( w_a+\frac{w_j}{\Delta t^2} \right) \ddot{s}_{0}^{2}+\left( w_a+w_{a,\mathrm{end}}+\frac{w_j}{\Delta t^2} \right) \ddot{s}_{n-1}^{2}-2w_{a,\mathrm{end}}\ddot{s}_{\mathrm{end}}\ddot{s}_{n-1}\\&+\left( w_a+\frac{2w_j}{\Delta t^2} \right) \sum_{i=1}^{n-2}{\ddot{s}_{i}^{2}}-\frac{2w_j}{\Delta t^2}\sum_{i=0}^{n-2}{\ddot{s}_i\ddot{s}_{i+1}}\end{aligned} J=wsi=0n2si2+ws,endsn122wsi=0n2si,refsi2ws,endsendsn1+wvi=0n2s˙i2+wv,ends˙n122wvi=0n2s˙i,refs˙i2wv,ends˙ends˙n1+(wa+Δt2wj)s¨02+(wa+wa,end+Δt2wj)s¨n122wa,ends¨ends¨n1+(wa+Δt22wj)i=1n2s¨i2Δt22wji=0n2s¨is¨i+1

从而得到核矩阵

P = [ P s P s ˙ P s ¨ ] 3 n × 3 n \boldsymbol{P}=\left[ \begin{matrix} \boldsymbol{P}_s& & \\ & \boldsymbol{P}_{\dot{s}}& \\ & & \boldsymbol{P}_{\ddot{s}}\\\end{matrix} \right] _{3n\times 3n} P=PsPs˙Ps¨3n×3n

和偏置向量

q = [ − 2 w s s 0 , r e f ⋮ − 2 w s s n − 2 , r e f − 2 w s , e n d s e n d − 2 w v s ˙ 0 , r e f ⋮ − 2 w v s ˙ n − 2 , r e f − 2 w v , e n d s ˙ e n d 0 ⋮ 0 − 2 w a , e n d s ¨ e n d ] 3 n × 1 \boldsymbol{q}=\left[ \begin{array}{c} -2w_ss_{0,\mathrm{ref}}\\ \vdots\\ -2w_ss_{n-2,\mathrm{ref}}\\ -2w_{s,\mathrm{end}}s_{\mathrm{end}}\\ \hline -2w_v\dot{s}_{0,\mathrm{ref}}\\ \vdots\\ -2w_v\dot{s}_{n-2,\mathrm{ref}}\\ -2w_{v,\mathrm{end}}\dot{s}_{\mathrm{end}}\\ \hline 0\\ \vdots\\ 0\\ -2w_{a,\mathrm{end}}\ddot{s}_{\mathrm{end}}\\\end{array} \right] _{3n\times 1} q=2wss0,ref2wssn2,ref2ws,endsend2wvs˙0,ref2wvs˙n2,ref2wv,ends˙end002wa,ends¨end3n×1

2.3 约束条件

根据运动学公式可得运动学等式约束

{ s ˙ i + 1 = s ˙ i + s ¨ i Δ t + 1 2 j i Δ t 2 = s ˙ i + 1 2 s ¨ i Δ t + 1 2 s ¨ i + 1 Δ t s i + 1 = s i + s ˙ i Δ t + 1 2 s ¨ i Δ t 2 + 1 6 j i Δ t 3 = s i + s ˙ i Δ t + 1 3 s ¨ i Δ t 2 + 1 6 s ¨ i + 1 Δ t 3 \begin{cases} \dot{s}_{i+1}=\dot{s}_i+\ddot{s}_i\Delta t+\frac{1}{2}j_i\Delta t^2=\dot{s}_i+\frac{1}{2}\ddot{s}_i\Delta t+\frac{1}{2}\ddot{s}_{i+1}\Delta t\\ s_{i+1}=s_i+\dot{s}_i\Delta t+\frac{1}{2}\ddot{s}_i\Delta t^2+\frac{1}{6}j_i\Delta t^3=s_i+\dot{s}_i\Delta t+\frac{1}{3}\ddot{s}_i\Delta t^2+\frac{1}{6}\ddot{s}_{i+1}\Delta t^3\\\end{cases} {s˙i+1=s˙i+s¨iΔt+21jiΔt2=s˙i+21s¨iΔt+21s¨i+1Δtsi+1=si+s˙iΔt+21s¨iΔt2+61jiΔt3=si+s˙iΔt+31s¨iΔt2+61s¨i+1Δt3

约束初始状态

s ˙ 0 = s ˙ i n i t , s ˙ 0 = s ˙ i n i t , s ¨ 0 = s ¨ i n i t \dot{s}_0=\dot{s}_{\mathrm{init}}, \dot{s}_0=\dot{s}_{\mathrm{init}}, \ddot{s}_0=\ddot{s}_{\mathrm{init}} s˙0=s˙init,s˙0=s˙init,s¨0=s¨init

此外还需保证各阶状态量满足不等式约束

s min ⁡ ⩽ s i ⩽ s max ⁡ , s ˙ min ⁡ ⩽ s ˙ i ⩽ s ˙ max ⁡ , s ¨ min ⁡ ⩽ s ¨ i ⩽ s ¨ max ⁡ , s ¨ min ⁡ ( 3 ) Δ t ⩽ s ¨ i + 1 − s ¨ i ⩽ s ¨ max ⁡ ( 3 ) Δ t s_{\min}\leqslant s_i\leqslant s_{\max}, \dot{s}_{\min}\leqslant \dot{s}_i\leqslant \dot{s}_{\max}, \ddot{s}_{\min}\leqslant \ddot{s}_i\leqslant \ddot{s}_{\max}, \ddot{s}_{\min}^{(3)}\Delta t\leqslant \ddot{s}_{i+1}-\ddot{s}_i\leqslant \ddot{s}_{\max}^{(3)}\Delta t sminsismax,s˙mins˙is˙max,s¨mins¨is¨max,s¨min(3)Δts¨i+1s¨is¨max(3)Δt

从而得到约束矩阵

A = [ A i n i t , 3 × 3 n A b o u n d , 3 n + ( n − 1 ) × 3 n A s y s , 2 ( n − 1 ) × 3 n ] 6 n × 3 n , l = [ l i n i t l b o u n d l s y s ] 6 n × 1 , u = [ u i n i t u b o u n d u s y s ] 6 n × 1 \boldsymbol{A}=\left[ \begin{array}{c} \boldsymbol{A}_{\mathrm{init}, 3\times 3n}\\ \boldsymbol{A}_{\mathrm{bound}, 3n+\left( n-1 \right) \times 3n}\\ \boldsymbol{A}_{\mathrm{sys}, 2\left( n-1 \right) \times 3n}\\\end{array} \right] _{6n\times 3n}, \boldsymbol{l}=\left[ \begin{array}{c} \boldsymbol{l}_{\mathrm{init}}\\ \boldsymbol{l}_{\mathrm{bound}}\\ \boldsymbol{l}_{\mathrm{sys}}\\\end{array} \right] _{6n\times 1}, \boldsymbol{u}=\left[ \begin{array}{c} \boldsymbol{u}_{\mathrm{init}}\\ \boldsymbol{u}_{\mathrm{bound}}\\ \boldsymbol{u}_{\mathrm{sys}}\\\end{array} \right] _{6n\times 1} A=Ainit,3×3nAbound,3n+(n1)×3nAsys,2(n1)×3n6n×3n,l=linitlboundlsys6n×1,u=uinituboundusys6n×1

2.4 二次规划形式

综上所述,可得到PJSO算法的二次规划形式

min ⁡ x J = x T P x + q T x s . t . l ⩽ A x ⩽ u \min _{\boldsymbol{x}}J=\boldsymbol{x}^T\boldsymbol{Px}+\boldsymbol{q}^T\boldsymbol{x}\\\mathrm{s}.\mathrm{t}. \boldsymbol{l}\leqslant \boldsymbol{Ax}\leqslant \boldsymbol{u} xminJ=xTPx+qTxs.t.lAxu

3 算法仿真

3.1 ROS C++仿真

核心代码如下所示:

bool PiecewiseJerkVelocityPlanner::plan(const Points3d& waypoints, VelocityProfile& velocity_profile)
{int dim = static_cast<int>(waypoints.size());std::vector<double> s_ref(dim, 0.0);for (int i = 1; i < dim; ++i){s_ref[i] =s_ref[i - 1] + std::hypot(waypoints[i].x() - waypoints[i - 1].x(), waypoints[i].y() - waypoints[i - 1].y());}x_max_ = s_ref.back();// construct QP ProblemEigen::MatrixXd P, A;std::vector<c_float> l, u, q;_calKernel(dim, P);_calOffset(dim, s_ref, q);_calAffineConstraint(dim, A);_calBoundary(dim, s_ref, l, u);std::vector<c_float> P_data;std::vector<c_int> P_indices;std::vector<c_int> P_indptr;int ind_P = 0;for (int col = 0; col < P.cols(); ++col){P_indptr.push_back(ind_P);for (int row = 0; row <= col; ++row){P_data.push_back(P(row, col));P_indices.push_back(row);ind_P++;}}P_indptr.push_back(ind_P);std::vector<c_float> A_data;std::vector<c_int> A_indices;std::vector<c_int> A_indptr;int ind_A = 0;for (int col = 0; col < A.cols(); ++col){A_indptr.push_back(ind_A);for (int row = 0; row < A.rows(); ++row){double data = A(row, col);if (std::fabs(data) > kMathEpsilon){A_data.push_back(data);A_indices.push_back(row);++ind_A;}}}A_indptr.push_back(ind_A);// solveOSQPWorkspace* work = nullptr;OSQPData* data = reinterpret_cast<OSQPData*>(c_malloc(sizeof(OSQPData)));OSQPSettings* settings = reinterpret_cast<OSQPSettings*>(c_malloc(sizeof(OSQPSettings)));osqp_set_default_settings(settings);settings->verbose = false;settings->warm_start = true;data->n = 3 * dim;data->m = 6 * dim;data->P = csc_matrix(data->n, data->n, P_data.size(), P_data.data(), P_indices.data(), P_indptr.data());data->A = csc_matrix(data->m, data->n, A_data.size(), A_data.data(), A_indices.data(), A_indptr.data());data->q = q.data();data->l = l.data();data->u = u.data();osqp_setup(&work, data, settings);osqp_solve(work);auto status = work->info->status_val;if ((status < 0) || (status != 1 && status != 2)){R_DEBUG << "failed optimization status: " << work->info->status;return false;}// parsefor (int i = 0; i < dim; ++i){velocity_profile.push(dt_ * i, work->solution->x[i], work->solution->x[dim + i], work->solution->x[2 * dim + i]);}// Cleanuposqp_cleanup(work);c_free(data->A);c_free(data->P);c_free(data);c_free(settings);return true;
}

这里设置初始速度 v 0 = 1.0 m / s v_0=1.0\ m/s v0=1.0 m/s,初始加速度 a 0 = 0 m / s 2 a_0=0\ m/s^2 a0=0 m/s2;终点速度 v 0 = 0.0 m / s v_0=0.0\ m/s v0=0.0 m/s,终点加速度 a 0 = 0 m / s 2 a_0=0\ m/s^2 a0=0 m/s2

在这里插入图片描述

3.2 Python仿真

核心代码如下所示:

def process(self, path: List[Point3d]) -> List[Dict]:velocity_profiles = []path_segs, path_refs = PathPlanner.getPathSegments(path)for i, path_seg in enumerate(path_segs):s_ref = path_refs[i]n = max(len(path_refs[i]),round(self.r * (self.dx_max ** 2 + s_ref[-1] * self.ddx_max) \/ (self.dx_max * self.ddx_max * self.dt)))s_ref = np.linspace(0, s_ref[-1], n)'''min x^T P x + q^T xs.t. l <= Ax <= u'''P = sparse.csc_matrix(self.calKernel(n))q = self.calOffset(n, s_ref)A = sparse.csc_matrix(self.calAffineConstraint(n))l, u = self.calBoundary(n, s_ref)solver = osqp.OSQP()solver.setup(P, q, A, l, u, verbose=False)res = solver.solve()sol = res.xvelocity_profiles.append({"time": [self.dt * i for i in range(n)],"path": PathPlanner.pathInterpolation(path_seg, n),"distance": sol[:n].tolist(),"velocity": sol[n:2 * n].tolist(),"acceleration": sol[2 * n:3 * n].tolist(),})return velocity_profiles

对于如下图所示的倒车路径

在这里插入图片描述

规划的速度和加速度曲线如下所示,中间速度为0处为换挡点

在这里插入图片描述


🔥 更多精彩专栏

  • 《ROS从入门到精通》
  • 《Pytorch深度学习实战》
  • 《机器学习强基计划》
  • 《运动规划实战精讲》

👇源码获取 · 技术交流 · 抱团学习 · 咨询分享 请联系👇

相关文章:

基于Piecewise Jerk Speed Optimizer的速度规划算法(附ROS C++/Python仿真)

目录 1 时空解耦运动规划2 PJSO速度规划原理2.1 优化变量2.2 代价函数2.3 约束条件2.4 二次规划形式 3 算法仿真3.1 ROS C仿真3.2 Python仿真 1 时空解耦运动规划 在自主移动系统的运动规划体系中&#xff0c;时空解耦的递进式架构因其高效性与工程可实现性被广泛采用。这一架…...

关于 JavaScript 版本、TypeScript、Vue 的区别说明, PHP 开发者入门 Vue 的具体方案

以下是关于 JavaScript 版本、TypeScript、Vue 的区别说明&#xff0c;以及 PHP 开发者入门 Vue 的具体方案&#xff1a; 一、JavaScript 版本演进 JavaScript 的核心版本以 ECMAScript 规范&#xff08;ES&#xff09; 命名&#xff1a; 版本发布时间关键特性ES52009严格模式…...

中断和信号详解

三种中断 中断分为三种&#xff1a;硬件中断、异常中断、软中断 硬件中断 设备向中断控制器发送中断请求&#xff0c;中断控制器生成对应中断号&#xff0c;然后通过中断引脚向cpu发送高电平&#xff0c;cpu收到请求后不会立即处理&#xff0c;cpu会处理完当前指令&#xff…...

STM32八股【10】-----stm32启动流程

启动流程 1.上电复位 2.系统初始化 3.跳转到 main 函数 启动入口&#xff1a; cpu被清空&#xff0c;程序从0x00000000开始运行0x00000000存放的是reset_handler的入口地址0x00000000的实际位置会变&#xff0c;根据不同的启动模式决定启动模式分为&#xff1a; flash启动&a…...

游戏引擎学习第312天:跨实体手动排序

运行游戏并评估当前状况 目前排序功能基本已经正常&#xff0c;能够实现特定的排序要求&#xff0c;针对单一区域、单个房间的场景&#xff0c;效果基本符合预期。 不过还有一些细节需要调试。现在有些对象的缩放比例不对&#xff0c;导致它们看起来有些怪异&#xff0c;需要…...

智警杯备赛--数据库管理与优化及数据库对象创建与管理

sql操作 插入数据 如果要操作数据表中的数据&#xff0c;首先应该确保表中存在数据。没有插入数据之前的表只是一张空表&#xff0c;需要使用insert语句向表中插入数据。插入数据有4种不同的方式&#xff1a;为所有字段插入数据、为指定字段插入数据、同时插入多条数据以及插…...

MySQL 在 CentOS 7 环境下的安装教程

&#x1f31f; 各位看官好&#xff0c;我是maomi_9526&#xff01; &#x1f30d; 种一棵树最好是十年前&#xff0c;其次是现在&#xff01; &#x1f680; 今天来学习Mysql的相关知识。 &#x1f44d; 如果觉得这篇文章有帮助&#xff0c;欢迎您一键三连&#xff0c;分享给更…...

K8S集群主机网络端口不通问题排查

一、环境&#xff1a; k8s: v1.23.6 docker: 20.10.14 问题和故障现象&#xff1a;devops主机集群主机节点到端口8082不通&#xff08;网络策略已经申请&#xff0c;并且网络策略已经实施完毕&#xff09;&#xff0c;而且网络实施人员再次确认&#xff0c;网络策…...

【Elasticsearch】retry_on_conflict

在 Elasticsearch 中&#xff0c;retry_on_conflict 是 _update 和 _update_by_query API 的一个参数&#xff0c;用于处理并发冲突。当多个客户端同时尝试更新同一个文档时&#xff0c;可能会发生版本冲突&#xff08;version conflict&#xff09;。retry_on_conflict 参数允…...

Android Cameara2 + MediaRecorder 完成录像功能

一、打开相机、预览 打开相机预览流程是Camera2的默认流程 可参考&#xff1a;https://blog.csdn.net/kk3087961/article/details/135616576 二、开启录像功能 开启录像主要包括以下3步&#xff1a; private void startRecording() {// 1. 停止预览并关闭会话if (mCameraSes…...

python打卡day39

知识点回顾 图像数据的格式&#xff1a;灰度和彩色数据模型的定义显存占用的4种地方 模型参数梯度参数优化器参数数据批量所占显存神经元输出中间状态 batchisize和训练的关系 课程代码&#xff1a; # 先继续之前的代码 import torch import torch.nn as nn import torch.opti…...

3.8.5 利用RDD统计网站每月访问量

本项目旨在利用Spark RDD统计网站每月访问量。首先&#xff0c;创建名为“SparkRDDWebsiteTraffic”的Maven项目&#xff0c;并添加Spark和Scala的依赖。接着&#xff0c;编写Scala代码&#xff0c;通过SparkContext读取存储在HDFS上的原始数据文件&#xff0c;使用map和reduce…...

尚硅谷redis7 49-51 redis管道之理论简介

前提redis事务和redis管道有点像&#xff0c;但本质上截然不同 49 redis管道之理论简介 面试题 如何优化频繁命令往返造成的性能瓶颈&#xff1f; redis每秒可以承受8万的写操作和接近10万次以上的读操作。每条命令都发送、处理、返回&#xff0c;能不能批处理一次性搞定呢…...

Spring Boot + MyBatis-Plus实现操作日志记录

创建数据库表 CREATE TABLE sys_operation_log (log_id bigint NOT NULL AUTO_INCREMENT COMMENT 日志ID,operation_type varchar(20) NOT NULL COMMENT 操作类型,operation_module varchar(50) NOT NULL COMMENT 操作模块,operation_desc varchar(200) DEFAULT NULL COMMENT …...

JavaScript入门基础篇-day03

一、为什么需要数组&#xff1f; 在我们正式学习数组之前&#xff0c;先思考一个场景&#xff1a;假设我们要记录一个班级50位同学的期末成绩。如果不用数组&#xff0c;代码会是这样的&#xff1a; let score1 85; let score2 92; let score3 78; // ... 要写50个变量&am…...

Leetcode-5 好数对的数目

Leetcode-5 好数对的数目&#xff08;简单&#xff09; 题目描述思路分析通过代码&#xff08;python&#xff09; 题目描述 给你一个整数数组 nums 。 如果一组数字 (i,j) 满足 nums[i] nums[j] 且 i < j &#xff0c;就可以认为这是一组 好数对 。 返回好数对的数目。 示…...

openEuler安装MySql8(tar包模式)

操作系统版本&#xff1a; openEuler release 22.03 (LTS-SP4) MySql版本&#xff1a; 下载地址&#xff1a; https://dev.mysql.com/downloads/mysql/ 准备安装&#xff1a; 关闭防火墙&#xff1a; 停止防火墙 #systemctl stop firewalld.service 关闭防火墙 #systemc…...

Opencv实用操作6 开运算 闭运算 梯度运算 礼帽 黑帽

1.相关函数 开运算 img_open cv2.morphologyEx(img,cv2.MORPH_OPEN,kernel)#&#xff08;图片&#xff0c;算法&#xff0c;核&#xff09; 闭运算 img_close cv2.morphologyEx(img,cv2.MORPH_CLOSE,kernel)#&#xff08;图片&#xff0c;算法&#xff0c;核&#xff09; 梯度…...

基于python,html,flask,echart,ids/ips,VMware,mysql,在线sdn防御ddos系统

详细视频:【基于python,html,flask,echart,ids/ips,VMware,mysql,在线sdn防御ddos系统-哔哩哔哩】 https://b23.tv/azUqQXe...

Git:现代软件开发的基石——原理、实践与行业智慧·优雅草卓伊凡

Git&#xff1a;现代软件开发的基石——原理、实践与行业智慧优雅草卓伊凡 一、Git的本质与核心原理 1. 技术定义 Git是一个分布式版本控制系统&#xff08;DVCS&#xff09;&#xff0c;由Linus Torvalds在2005年为管理Linux内核开发而创建。其核心是通过快照&#xff08;Sna…...

NLua性能对比:C#注册函数 vs 纯Lua实现

引言 在NLua开发中&#xff0c;我们常面临一个重要选择&#xff1a;将C#函数注册到Lua环境调用&#xff0c;还是直接在Lua中实现逻辑&#xff1f; 直觉告诉我们&#xff0c;C#作为编译型语言性能更高&#xff0c;但跨语言调用的开销是否会影响整体性能&#xff1f;本文通过基准…...

【计算机网络】第2章:应用层—Web and HTTP

目录 一、Web 与 HTTP 二、总结 &#xff08;一&#xff09;Web 的定义与功能 &#xff08;二&#xff09;HTTP 协议的定义与功能 &#xff08;三&#xff09;HTTP 协议的核心机制 1. HTTP 请求与响应流程 2. HTTP 的连接类型 3. HTTP 的状态码 &#xff08;四&#xf…...

HarmonyOS 5 应用开发导读:从入门到实践

一、HarmonyOS 5 概述 HarmonyOS 5 是华为推出的新一代分布式操作系统&#xff0c;其核心设计理念是"一次开发&#xff0c;多端部署"。与传统的移动操作系统不同&#xff0c;HarmonyOS 5 提供了更强大的跨设备协同能力&#xff0c;支持手机、平板、智能穿戴、智慧屏…...

大数据治理:分析中的数据安全

引言 随着大数据技术在各行业的深度应用&#xff0c;海量数据蕴含的价值被不断挖掘。然而&#xff0c;数据规模的爆发式增长与分析场景的复杂化&#xff0c;使数据安全问题日益凸显。从数据泄露、隐私侵犯到非法访问&#xff0c;每一个安全漏洞都可能带来难以估量的损失。本文将…...

数字孪生技术赋能西门子安贝格工厂:全球智能制造标杆的数字化重构实践

在工业4.0浪潮席卷全球制造业的当下&#xff0c;西门子安贝格电子制造工厂&#xff08;Electronic Works Amberg, EWA&#xff09;凭借数字孪生技术的深度应用&#xff0c;构建起全球制造业数字化转型的典范。这座位于德国巴伐利亚州的“未来工厂”&#xff0c;通过虚实融合的数…...

国内高频混压PCB厂家有哪些?

一、技术领先型厂商&#xff08;聚焦材料与工艺突破&#xff09; 猎板PCB 技术亮点&#xff1a;真空层压工艺实现FR-4与罗杰斯高频材料&#xff08;RO4350B/RO3003&#xff09;混压&#xff0c;阻抗公差3%&#xff0c;支持64单元/板的5G天线模块&#xff0c;插损降低15%。 应用…...

【图像处理基石】立体匹配的经典算法有哪些?

1. 立体匹配的经典算法有哪些&#xff1f; 立体匹配是计算机视觉中从双目图像中获取深度信息的关键技术&#xff0c;其经典算法按技术路线可分为以下几类&#xff0c;每类包含若干代表性方法&#xff1a; 1.1 基于区域的匹配算法&#xff08;Local Methods&#xff09; 通过…...

day12 leetcode-hot100-19(矩阵2)

54. 螺旋矩阵 - 力扣&#xff08;LeetCode&#xff09; 1.模拟路径 思路&#xff1a;模拟旋转的路径 &#xff08;1&#xff09;设计上下左右方向控制器以及边界。比如zy1向右&#xff0c;zy-1向左&#xff1b;sx1向上&#xff0c;sx-1向下。上边界0&#xff0c;下边界hang-1&a…...

将Java应用集成到CI/CD管道:从理论到生产实践

在2025年的软件开发领域&#xff0c;持续集成与持续部署&#xff08;CI/CD&#xff09;已成为敏捷开发和DevOps的核心实践。根据2024年DevOps报告&#xff0c;85%的企业通过CI/CD管道实现了交付周期缩短50%以上&#xff0c;特别是在金融、电商和SaaS行业。Java&#xff0c;作为…...

密钥管理系统在存储加密场景中的深度实践:以TDE透明加密守护文件服务器安全

引言&#xff1a;数据泄露阴影下的存储加密革命 在数字化转型的深水区&#xff0c;企业数据资产正面临前所未有的安全挑战。据IBM《2025年数据泄露成本报告》显示&#xff0c;全球单次数据泄露事件平均成本已达465万美元&#xff0c;其中存储介质丢失或被盗导致的损失占比高达…...