上线了免费建网站/谷歌推广代理公司

上线了免费建网站,谷歌推广代理公司,衡阳做网站ss0734,东莞企业网站建设文章目录 前言1. 变量定义2. 摇杆死区设置3. 模式检查4. 摇杆数据处理4.1 右摇杆垂直值(psx_buf[7])4.2 右摇杆水平值(psx_buf[8])4.3 左摇杆水平值(psx_buf[5])4.4 左摇杆垂直值(psx_buf[6]&am…

文章目录

  • 前言
  • 1. 变量定义
  • 2. 摇杆死区设置
  • 3. 模式检查
  • 4. 摇杆数据处理
    • 4.1 右摇杆垂直值(psx_buf[7])
    • 4.2 右摇杆水平值(psx_buf[8])
    • 4.3 左摇杆水平值(psx_buf[5])
    • 4.4 左摇杆垂直值(psx_buf[6])
  • 5. 电机转速更新
  • 6. 保存当前转速值
  • 7. 原理总结
  • 8. 参数转化示例
    • 假设
    • 计算
    • 最终电机转速


前言

这段代码的作用是通过PS2手柄的摇杆数据来控制小车的四个电机转速。下面我将详细解释这段代码的逻辑、PS2手柄的参数如何转化为小车转速,以及其背后的原理。


1. 变量定义

float car_left1=0, car_right1=0, car_left2=0, car_right2=0;
static float car_left1_bak = 0, car_right1_bak = 0, car_left2_bak = 0, car_right2_bak = 0;

car_left1, car_right1, car_left2, car_right2:分别表示小车四个电机的目标转速。
car_left1_bak, car_right1_bak, car_left2_bak, car_right2_bak:用于保存上一次的电机转速值,用于判断是否需要更新电机转速。

2. 摇杆死区设置

static u8 num = 10;

num 是一个阈值,用于设置摇杆的“死区”。当摇杆的值变化小于 num 时,忽略这些微小的变化,避免电机频繁调整。

摇杆的原始值范围是 0~255中间值是 127。如果摇杆的值与 127 的差值小于 num,则认为摇杆处于“中立”位置。

3. 模式检查

if (psx_buf[1] != PS2_LED_RED)return;

psx_buf[1] 是PS2手柄的当前模式(例如红灯模式或绿灯模式)。
如果当前模式不是红灯模式(PS2_LED_RED),则直接返回,不处理摇杆数据

4. 摇杆数据处理

PS2手柄的摇杆数据存储在 psx_buf 数组中:
psx_buf[5]:左摇杆的水平值(X轴)。
psx_buf[6]:左摇杆的垂直值(Y轴)。
psx_buf[7]:右摇杆的水平值(X轴)。
psx_buf[8]:右摇杆的垂直值(Y轴)。
摇杆值的范围是 0~255,中间值是 127。通过计算摇杆值与 127 的差值,可以确定摇杆的偏移方向和幅度。

4.1 右摇杆垂直值(psx_buf[7])

if (abs_int(127 - psx_buf[7]) > num)
{car_left1 = car_left1 - (0x7f - psx_buf[7]) * 2;car_right1 = car_right1 + (0x7f - psx_buf[7]) * 2;car_left2 = car_left2 - (0x7f - psx_buf[7]) * 2;car_right2 = car_right2 + (0x7f - psx_buf[7]) * 2;
}

右摇杆的垂直值控制小车的前后运动。
(0x7f - psx_buf[7]):计算摇杆值与中立值 127 的差值。
如果摇杆向上推(psx_buf[7] < 127),差值为正,小车向前运动。
如果摇杆向下拉(psx_buf[7] > 127),差值为负,小车向后运动。
将差值放大,增加电机转速的灵敏度。
通过调整 car_left1, car_right1, car_left2, car_right2 的值,控制四个电机的转速。

4.2 右摇杆水平值(psx_buf[8])

if (abs_int(127 - psx_buf[8]) > num)
{car_left1 = car_left1 + (0x7f - psx_buf[8]) * 2;car_right1 = car_right1 + (0x7f - psx_buf[8]) * 2;car_left2 = car_left2 + (0x7f - psx_buf[8]) * 2;car_right2 = car_right2 + (0x7f - psx_buf[8]) * 2;
}

右摇杆的水平值控制小车的左右平移。
(0x7f - psx_buf[8]):计算摇杆值与中立值 127 的差值。
如果摇杆向右推(psx_buf[8] > 127),差值为负,小车向右平移。
如果摇杆向左推(psx_buf[8] < 127),差值为正,小车向左平移。
将差值放大,增加电机转速的灵敏度。

4.3 左摇杆水平值(psx_buf[5])

if (abs_int(127 - psx_buf[5]) > num)
{car_left1 = car_left1 - (0x7f - psx_buf[5]) * 2;car_right1 = car_right1 + (0x7f - psx_buf[5]) * 2;car_left2 = car_left2 + (0x7f - psx_buf[5]) * 2;car_right2 = car_right2 - (0x7f - psx_buf[5]) * 2;
}

左摇杆的水平值控制小车的旋转。
(0x7f - psx_buf[5]):计算摇杆值与中立值 127 的差值。
如果摇杆向右推(psx_buf[5] > 127),差值为负,小车顺时针旋转。
如果摇杆向左推(psx_buf[5] < 127),差值为正,小车逆时针旋转。
将差值放大,增加电机转速的灵敏度。

4.4 左摇杆垂直值(psx_buf[6])

if (abs_int(127 - psx_buf[6]) > num)
{car_left1 = car_left1 + (0x7f - psx_buf[6]) * 2;car_right1 = car_right1 + (0x7f - psx_buf[6]) * 2;car_left2 = car_left2 + (0x7f - psx_buf[6]) * 2;car_right2 = car_right2 + (0x7f - psx_buf[6]) * 2;
}

左摇杆的垂直值控制小车的前后运动(与右摇杆垂直值类似)。
(0x7f - psx_buf[6]):计算摇杆值与中立值 127 的差值。
如果摇杆向上推(psx_buf[6] < 127),差值为正,小车向前运动。
如果摇杆向下拉(psx_buf[6] > 127),差值为负,小车向后运动。
将差值放大,增加电机转速的灵敏度。

5. 电机转速更新

if ((car_left1_bak != car_left1) || (car_right1_bak != car_right1) || (car_left2_bak != car_left2) || (car_right2_bak != car_right2))
{motor_speed_set(car_left1 / 1000, car_right1 / 1000, car_left2 / 1000, car_right2 / 1000);
}

如果当前计算的电机转速与上一次保存的值不同,则调用 motor_speed_set 函数更新电机转速。
car_left1 / 1000:将转速值缩小,可能是为了适配电机控制器的输入范围。

6. 保存当前转速值

car_left1_bak = car_left1;
car_right1_bak = car_right1;
car_left2_bak = car_left2;
car_right2_bak = car_right2;

保存当前计算的电机转速值,用于下一次比较。

7. 原理总结

  1. PS2手柄的摇杆值通过计算与中立值 127 的差值,确定摇杆的偏移方向和幅度
  2. 根据摇杆的偏移方向和幅度,调整四个电机的转速,实现小车的前后、左右、旋转等运动。
  3. 通过设置死区(num),避免摇杆微小变化导致电机频繁调整。
  4. 最终通过 motor_speed_set 函数将计算出的转速值传递给电机控制器,实现小车的运动控制。

8. 参数转化示例

假设

psx_buf[7] = 100(右摇杆向上推)。
psx_buf[8] = 150(右摇杆向右推)。
psx_buf[5] = 80(左摇杆向左推)。
psx_buf[6] = 127(左摇杆中立)。

计算

右摇杆垂直值:127 - 100 = 27,差值为正,小车向前运动。
右摇杆水平值:127 - 150 = -23,差值为负,小车向右平移。
左摇杆水平值:127 - 80 = 47,差值为正,小车逆时针旋转。
左摇杆垂直值:127 - 127 = 0,无变化。

最终电机转速

car_left1 = 0 - 272 + (-23)2 - 472 = -194
car_right1 = 0 + 27
2 + (-23)2 + 472 = 102
car_left2 = 0 - 272 + (-23)2 + 472 = -6
car_right2 = 0 + 27
2 + (-23)2 - 472 = -86

通过 motor_speed_set 函数将这些值传递给电机控制器,实现小车的运动。


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

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

相关文章

阿里千问大模型(Qwen2.5-VL-7B-Instruct)部署

参考链接 知乎帖子 B站视频 huggingface 镜像网站&#xff08;不太全&#xff0c;比如 Qwen/Qwen2.5-VL-7B-Instruct就没有&#xff09; huggingface 5种下载方式汇总 通过huggingface-cli下载模型 不一样的部分是预训练权重的下载和demo 首先安装huggingface_hub pip insta…

Jenkins在Windows上的使用(二):自动拉取、打包、部署

&#xff08;一&#xff09;Jenkins全局配置 访问部署好的Jenkins服务器网址localhost:8080&#xff0c;完成默认插件的安装后&#xff0c;接下来将使用SSH登录远程主机以实现自动化部署。 1. 配置插件 选择dashboard->Manage Jenkins->plugins 安装下面两个插件  …

群晖DS 223 Docker:开启私有云

群晖DS 223 Docker&#xff1a;开启私有云的无限可能 引言 在数据存储与管理的不断演进中&#xff0c;群晖 DS 223 凭借其出色的性能和丰富的功能&#xff0c;成为众多用户搭建私有云的热门选择。而当它与 Docker 技术相遇&#xff0c;犹如为数据管理的舞台添上了绚丽多彩的灯…

Three.js 进阶(灯光阴影关系和设置、平行光、阴影相机)

本篇主要学习内容 : 灯光与阴影聚光灯点光源平行光阴影相机和阴影计算投射阴影接受阴影 点赞 关注 收藏 学会了 1.灯光与阴影 1、材质要满足能够对光有反应 2、设置渲染器开启阴影计算 renderer.shadowMap.enabledtrue 3、设置光照投射阴影 directionalLight.castShadow …

【 <一> 炼丹初探:JavaWeb 的起源与基础】之 Tomcat 的工作原理:从启动到请求处理的流程

<前文回顾> 点击此处查看 合集 https://blog.csdn.net/foyodesigner/category_12907601.html?fromshareblogcolumn&sharetypeblogcolumn&sharerId12907601&sharereferPC&sharesourceFoyoDesigner&sharefromfrom_link <今日更新> 一、Tomcat…

计网面试准备

正确理解网络数据传输过程 同一路由器的不同接口属于不同局域网&#xff0c;广播只能在同一个局域网

【技术白皮书】内功心法 | 第二部分 | Telnet远程登录的工作原理

远程登录的工作原理 背景介绍远程登录远程登录的服务模式远程登录服务的实现基础远程登录服务的运行模式Telnet服务为什么不被操作系统管理 Telnet协议的原理网络虚终端&#xff08;NVT&#xff09;结束标示NVT的原理NVT屏蔽差异 背景介绍 绝大多数计算机都是运行多用户操作系…

游戏引擎学习第150天

回顾与当天计划 我们在这里完全不使用任何库&#xff0c;所以我们完全是引擎和库免疫的, 正如大家所知道的&#xff0c;我们正在编写自己的资源处理系统&#xff0c;准确来说&#xff0c;是一个资源加载系统。过去一周我们已经完成了很多工作&#xff0c;现在只剩下最后几步&a…

Flutter中stream学习

Flutter中stream学习 概述Stream的基础概念stream的常用方法Stream.fromFuture(Future<T> future)Stream.fromFutures(Iterable<Future<T>> futures)Stream.fromIterable(Iterable<T> elements)Stream.periodic(Duration period, [T computation(int c…

基于javaweb的SSM房屋租赁管理系统设计和实现(源码+文档+部署讲解)

技术范围&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬虫、数据可视化、小程序、安卓app、大数据、物联网、机器学习等设计与开发。 主要内容&#xff1a;免费功能设计、开题报告、任务书、中期检查PPT、系统功能实现、代码编写、论文编写和辅导、论…

物联网商业模式

物联网商业模式是一种战略规划&#xff0c;它融合了物联网技术来创造价值并获取收入。它与传统商业模式的不同之处在于&#xff0c;它利用互联设备来改善运营、提升客户体验以及优化服务项目。在当今由科技驱动的世界中&#xff0c;这种商业模式通过利用实时数据来提供创新服务…

从0开始的操作系统手搓教程45——实现exec

目录 建立抽象 实现加载 实现sys_execv &#xff01;&#xff01;&#xff01;提示&#xff1a;因为实现问题没有测试。所以更像是笔记&#xff01; exec 函数的作用是用新的可执行文件替换当前进程的程序体。具体来说&#xff0c;exec 会将当前正在运行的用户进程的进程体&…

【python爬虫】酷狗音乐爬取练习

注意&#xff1a;本次爬取的音乐仅有1分钟试听&#xff0c;仅作学习爬虫的原理&#xff0c;完整音乐需要自行下载客户端。 一、 初步分析 登陆酷狗音乐后随机选取一首歌&#xff0c;在请求里发现一段mp3文件&#xff0c;复制网址&#xff0c;确实是我们需要的url。 复制音频的…

Linux开发工具----vim

目录 Linux编辑器-vim使用 1. vim的基本概念 正常/普通/命令模式(Normal mode) 插入模式(Insert mode) 底行模式(last line mode) 2. vim的基本操作 3. vim正常模式命令集 4. vim底行模式命令集 5. vim操作总结 (本篇文章相当于vim常用命令字典) Linux编辑器-vim使用 我们先来看…

基于云函数的自习室预约微信小程序+LW示例参考

全阶段全种类学习资源&#xff0c;内涵少儿、小学、初中、高中、大学、专升本、考研、四六级、建造师、法考、网赚技巧、毕业设计等&#xff0c;持续更新~ 文章目录 [TOC](文章目录) 1.项目介绍2.项目部署3.项目部分截图4.获取方式 1.项目介绍 技术栈工具&#xff1a;云数据库…

卷积神经网络与计算机视觉:从数学基础到实战应用

卷积神经网络与计算机视觉&#xff1a;从数学基础到实战应用 摘要 本文深入解析卷积神经网络&#xff08;CNN&#xff09;的核心原理及其在计算机视觉中的应用。首先介绍卷积与互相关的数学定义及在神经网络中的实际应用差异&#xff0c;接着从系统设计视角分析卷积的线性代数…

从Manus到OpenManus:多智能体协作框架如何重构AI生产力?

文章目录 Manus&#xff1a;封闭生态下的通用AI智能体OpenManus&#xff1a;开源社区的闪速复刻挑战与未来&#xff1a;框架落地的现实边界当前局限性未来演进方向 OpenManus使用指南1. 环境配置2. 参数配置3. 替换搜索引擎4. 运行效果 协作框架开启AI生产力革命 Manus&#xf…

js 使用 Web Workers 来实现一个精确的倒计时,即使ios手机锁屏或页面进入后台,倒计时也不会暂停。

## 效果如上 <!-- 将 main.js 和 worker.js 放在同一个目录下&#xff0c;然后在 HTML 文件中引入 main.js --><!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content&q…

Docker Compose国内镜像一键部署dify

克隆代码 git clone https://github.com/langgenius/dify.git进入docker目录 cd docker修改.env部分 # 将环境模版文件变量重命名 cp .env.example .env # 修改 .env,修改nginx的host和端口,避免端口冲突 NGINX_SERVER_NAME192.168.1.223 NGINX_PORT1880 NGINX_SSL_PORT1443…

NO.29十六届蓝桥杯备战|string九道练习|reverse|翻转|回文(C++)

P5015 [NOIP 2018 普及组] 标题统计 - 洛谷 #include <bits/stdc.h> using namespace std;int main() {ios::sync_with_stdio(false);cin.tie(nullptr);string s;getline(cin, s);int sz s.size();int cnt 0;for (int i 0; i < sz; i){if (isspace(s[i]))continue…