【毕设项目】 基于stm32的四轴飞行器设计 - 物联网 嵌入式 单片机

article/2025/4/8 8:22:28

1 简介

Hi,大家好,这里是丹成学长,今天向大家介绍一个学长做的单片机项目

基于stm32的四轴飞行器设计

大家可用于 课程设计 或 毕业设计


单片机-嵌入式毕设选题大全及项目分享:

https://blog.csdn.net/m0_71572576/article/details/125409052


在这里插入图片描述

这次尝试制作一个四旋翼飞控的过程

这个飞控是基于STM32,整合了MPU6050,即陀螺仪和重力加速计,但没有融合电子罗盘;

这是飞控程序的控制流程(一个执行周期):

在这里插入图片描述

2 重点内容

2.1 i2c通信

通过GPIO模拟i2c,这样也能获得mpu6050的数据,虽然代码多了一些,但是比较好的理解i2c的原理。

STM32库实现的模拟i2c代码(注释好像因为编码问题跪了):

/*******************************************************************************
// file                :    i2c_conf.h
// MCU                : STM32F103VET6
// IDE                : Keil uVision4
// date                £º2014.2.28
*******************************************************************************/
#include "stm32f10x.h"#define   uchar unsigned char
#define   uint unsigned int#define      FALSE 0  
#define   TRUE  1void I2C_GPIO_Config(void);
void I2C_delay(void);
void delay5ms(void);
int I2C_Start(void);
void I2C_Stop(void);
void I2C_Ack(void);
void I2C_NoAck(void);
int I2C_WaitAck(void);
void I2C_SendByte(u8 SendByte);
unsigned char I2C_RadeByte(void);
int Single_Write(uchar SlaveAddress,uchar REG_Address,uchar REG_data);
unsigned char Single_Read(unsigned char SlaveAddress,unsigned char REG_Address);
/*******************************************************************************
// file                :  i2c_conf.c
// MCU                : STM32F103VET6
// IDE                : Keil uVision4
// date                £º2014.2.28
*******************************************************************************/#include "i2c_conf.h"#define SCL_H         GPIOB->BSRR = GPIO_Pin_6       
#define SCL_L         GPIOB->BRR  = GPIO_Pin_6         
#define SDA_H         GPIOB->BSRR = GPIO_Pin_7
#define SDA_L         GPIOB->BRR  = GPIO_Pin_7#define SCL_read      GPIOB->IDR  & GPIO_Pin_6        //IDR:¶Ë¿ÚÊäÈë¼Ä´æÆ÷¡£
#define SDA_read      GPIOB->IDR  & GPIO_Pin_7void I2C_GPIO_Config(void)
{GPIO_InitTypeDef  GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_6;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;   //¿ªÂ©Êä³öģʽGPIO_Init(GPIOB, &GPIO_InitStructure);GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_7;GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;GPIO_Init(GPIOB, &GPIO_InitStructure);
}void I2C_delay(void)
{int i=6; //ÕâÀï¿ÉÒÔÓÅ»¯ËÙ¶È    £¬¾­²âÊÔ×îµÍµ½5»¹ÄÜдÈëwhile(i) { i--; }  
}void delay5ms(void)
{int i=5000;  while(i) { i--; }  
}int I2C_Start(void)
{SDA_H;                                            //II2ЭÒé¹æ¶¨±ØÐëÔÚʱÖÓÏßΪµÍµçƽµÄÇ°ÌáÏ£¬²Å¿ÉÒÔÈà Êý¾ÝÏßÐźŸıäSCL_H;I2C_delay();if(!SDA_read)return FALSE;                //SDAÏßΪµÍµçƽÔò×ÜÏßæ,Í˳öSDA_L;                                I2C_delay();if(SDA_read) return FALSE;                //SDAÏßΪ¸ßµçƽÔò×ÜÏß³ö´í,Í˳öSDA_L;                                I2C_delay();                    return TRUE;
}void I2C_Stop(void)
{SCL_L;I2C_delay();SDA_L;I2C_delay();SCL_H;I2C_delay();SDA_H;I2C_delay();
} void I2C_Ack(void)
{    SCL_L;I2C_delay();SDA_L;I2C_delay();SCL_H;I2C_delay();SCL_L;I2C_delay();
}   void I2C_NoAck(void)
{    SCL_L;I2C_delay();SDA_H;I2C_delay();SCL_H;I2C_delay();SCL_L;I2C_delay();
} int I2C_WaitAck(void)      //·µ»ØΪ:=1ÓÐACK,  =0ÎÞACK
{SCL_L;I2C_delay();SDA_H;            I2C_delay();SCL_H;I2C_delay();if(SDA_read){SCL_L;I2C_delay();return FALSE;}SCL_L;I2C_delay();return TRUE;
}void I2C_SendByte(u8 SendByte) //Êý¾Ý´Ó¸ßλµ½µÍλ//
{u8 i=8;while(i--){SCL_L;I2C_delay();if(SendByte&0x80)        // 0x80 = 1000 0000;SDA_H;  else SDA_L;   SendByte<<=1;   // SendByte×óÒÆһλ¡£I2C_delay();SCL_H;I2C_delay();}SCL_L;
}  unsigned char I2C_RadeByte(void)  //Êý¾Ý´Ó¸ßλµ½µÍλ//
{ u8 i=8;u8 ReceiveByte=0;SDA_H;                while(i--){ReceiveByte<<=1;      //×óÒÆһ룬SCL_L;I2C_delay();SCL_H;I2C_delay();    if(SDA_read){ReceiveByte|=0x01; //дÈë}}SCL_L;return ReceiveByte;
} int Single_Write(uchar SlaveAddress,uchar REG_Address,uchar REG_data)             
{if(!I2C_Start())return FALSE;I2C_SendByte(SlaveAddress);   //·¢ËÍÉ豸µØÖ·+дÐźŠ   //I2C_SendByte(((REG_Address & 0x0700) >>7) | SlaveAddress & 0xFFFE);    //ÉèÖøßÆðʼµØÖ·+Æ÷¼þµØÖ· if(!I2C_WaitAck()){I2C_Stop(); return FALSE;}I2C_SendByte(REG_Address );   //ÉèÖõÍÆðʼµØÖ·      I2C_WaitAck();    I2C_SendByte(REG_data);I2C_WaitAck();   I2C_Stop(); delay5ms();return TRUE;
}unsigned char Single_Read(unsigned char SlaveAddress,unsigned char REG_Address)
{   unsigned char REG_data; if(!I2C_Start())return FALSE;I2C_SendByte(SlaveAddress); //I2C_SendByte(((REG_Address & 0x0700) >>7) | REG_Address & 0xFFFE);//ÉèÖøßÆðʼµØÖ·+Æ÷¼þµØÖ· if(!I2C_WaitAck()){I2C_Stop();return FALSE;}I2C_SendByte((u8) REG_Address);   //ÉèÖõÍÆðʼµØÖ·      I2C_WaitAck();I2C_Start();I2C_SendByte(SlaveAddress+1);I2C_WaitAck();REG_data= I2C_RadeByte();I2C_NoAck();I2C_Stop();return REG_data;}

2.2 mpu6050;

用写好的模拟i2c函数读取mpu6050,根据mpu6050手册的各寄存器地址,读取到了重力加速计和陀螺仪的各分量;

传感器采样率设置为200Hz,这个值是因为我电调频率为200Hz,也就是说,我的程序循环一次0.005s,一般来说,采样率高点没问题,别比执行一次闭环控制的周期长就行了;

陀螺仪量程±2000°/s,加速计量程±2g, 量程越大,取值越不精确;

这里注意,由于我们没有采用磁力计,而陀螺仪存在零偏,所以最终在yaw方向上没有绝对的参考系,不能建立绝对的地理坐标系,这样最好的结果也仅仅是在yaw上存在缓慢漂移。

2.3 互补滤波;

融合时,陀螺仪的积分运算很大程度上决定了飞行器的瞬时运动情况,而重力加速计通过长时间的累积不断矫正陀螺仪产生的误差,最终得到准确的机体姿态。

这里我采用Madgwick提供的UpdateIMU算法来得到姿态角所对应的四元数,之后只需要经过简单运算便可转换为实时欧拉角。

#define sampleFreq    512.0f        // sample frequency in Hz
#define betaDef        0.1f        // 2 * proportional gainvolatile float beta = betaDef;
volatile float q0 = 1.0f, q1 = 0.0f, q2 = 0.0f, q3 = 0.0f;float invSqrt(float x);void MadgwickAHRSupdateIMU(float gx, float gy, float gz, float ax, float ay, float az) {float recipNorm;float s0, s1, s2, s3;float qDot1, qDot2, qDot3, qDot4;float _2q0, _2q1, _2q2, _2q3, _4q0, _4q1, _4q2 ,_8q1, _8q2, q0q0, q1q1, q2q2, q3q3;// Rate of change of quaternion from gyroscopeqDot1 = 0.5f * (-q1 * gx - q2 * gy - q3 * gz);qDot2 = 0.5f * (q0 * gx + q2 * gz - q3 * gy);qDot3 = 0.5f * (q0 * gy - q1 * gz + q3 * gx);qDot4 = 0.5f * (q0 * gz + q1 * gy - q2 * gx);// Compute feedback only if accelerometer measurement valid (avoids NaN in accelerometer normalisation)if(!((ax == 0.0f) && (ay == 0.0f) && (az == 0.0f))) {// Normalise accelerometer measurementrecipNorm = invSqrt(ax * ax + ay * ay + az * az);ax *= recipNorm;ay *= recipNorm;az *= recipNorm;// Auxiliary variables to avoid repeated arithmetic_2q0 = 2.0f * q0;_2q1 = 2.0f * q1;_2q2 = 2.0f * q2;_2q3 = 2.0f * q3;_4q0 = 4.0f * q0;_4q1 = 4.0f * q1;_4q2 = 4.0f * q2;_8q1 = 8.0f * q1;_8q2 = 8.0f * q2;q0q0 = q0 * q0;q1q1 = q1 * q1;q2q2 = q2 * q2;q3q3 = q3 * q3;// Gradient decent algorithm corrective steps0 = _4q0 * q2q2 + _2q2 * ax + _4q0 * q1q1 - _2q1 * ay;s1 = _4q1 * q3q3 - _2q3 * ax + 4.0f * q0q0 * q1 - _2q0 * ay - _4q1 + _8q1 * q1q1 + _8q1 * q2q2 + _4q1 * az;s2 = 4.0f * q0q0 * q2 + _2q0 * ax + _4q2 * q3q3 - _2q3 * ay - _4q2 + _8q2 * q1q1 + _8q2 * q2q2 + _4q2 * az;s3 = 4.0f * q1q1 * q3 - _2q1 * ax + 4.0f * q2q2 * q3 - _2q2 * ay;recipNorm = invSqrt(s0 * s0 + s1 * s1 + s2 * s2 + s3 * s3); // normalise step magnitudes0 *= recipNorm;s1 *= recipNorm;s2 *= recipNorm;s3 *= recipNorm;// Apply feedback stepqDot1 -= beta * s0;qDot2 -= beta * s1;qDot3 -= beta * s2;qDot4 -= beta * s3;}// Integrate rate of change of quaternion to yield quaternionq0 += qDot1 * (1.0f / sampleFreq);q1 += qDot2 * (1.0f / sampleFreq);q2 += qDot3 * (1.0f / sampleFreq);q3 += qDot4 * (1.0f / sampleFreq);// Normalise quaternionrecipNorm = invSqrt(q0 * q0 + q1 * q1 + q2 * q2 + q3 * q3);q0 *= recipNorm;q1 *= recipNorm;q2 *= recipNorm;q3 *= recipNorm;
}float invSqrt(float x) {float halfx = 0.5f * x;float y = x;long i = *(long*)&y;i = 0x5f3759df - (i>>1);y = *(float*)&i;y = y * (1.5f - (halfx * y * y));return y;
}

2.4 获取期望姿态;

也就是遥控部分了,让用户介入控制。

在这里插入图片描述
通过HC-06蓝牙模块接连到STM32串口1,再无线连接到控制端,这样我们就可以获得控制端不断发送的数据包了,并实时更新期望姿态角,这里只需要注意输出的姿态角和实时姿态角方向一致以及数据包的校验就行了。

2.5 PID控制算法;

由于简单的线性控制不可能满足四轴飞行器这个灵敏的系统,引入PID控制器来更好的纠正系统。

简介:PID实指“比例proportional”、“积分integral”、“微分derivative”,这三项构成PID基本要素。每一项完成不同任务,对系统功能产生不同的影响。
在这里插入图片描述
以Pitch为例:

在这里插入图片描述

error为期望角减去实时角度得到的误差;

iState为积分i参数对应累积过去时间里的误差总和;

if语句限定iState范围,繁殖修正过度;

微分d参数为当前姿态减去上次姿态,估算当前速度(瞬间速度);

总调整量为p,i,d三者之和;

这样,P代表控制系统的响应速度,越大,响应越快。

I,用来累积过去时间内的误差,修正P无法达到的期望姿态值(静差);

D,加强对机体变化的快速响应,对P有抑制作用。

PID各参数的整定需要综合考虑控制系统的各个方面,才能达到最佳效果。

2.6 输出PWM信号

PID计算完成之后,便可以通过STM32自带的定时资源很容易的调制出四路pwm信号,采用的电调pwm格式为50Hz,高电平持续时间0.5ms-2.5ms;

我以1.0ms-2.0ms为每个电机的油门行程,这样,1ms的宽度均匀的对应电调的从最低到最高转速。

至此,一个用stm32和mpu6050搭建的飞控系统就算实现了。

剩下的调试PID参数了。


单片机-嵌入式毕设选题大全及项目分享:

https://blog.csdn.net/m0_71572576/article/details/125409052


6 最后


http://chatgpt.dhexx.cn/article/GlaCa6lg.shtml

相关文章

【毕业设计】基于 stm32 的病房呼叫系统 - 物联网 嵌入式 单片机

文章目录 1 简介2 绪论2.1 课题背景 3 系统设计3.1 系统架构3.2 主程序设计3.3 语音模块程序设计3.4 显示模块程序设计3.5 键盘模块程序设计3.6 无线传输模块程序设计3.7 部分实现代码 4 最后 1 简介 Hi&#xff0c;大家好&#xff0c;这里是丹成学长&#xff0c;今天向大家介…

嵌入式物联网软件开发实战系列(STM32+FreeRTOS)

一、使用STM32CubeMx 创建物联网项目软件工程 二、STM32CubeMx 配置时钟 三、STM32CubeMx 配置GPIO 及 GPIO 相关接口在软件工程中的形式及使用 四、STM32CubeMx 配置USART 串口 五、USART 串口在物联网项目中的使用之不定长数据接收&#xff08;DMA空闲中断方式&#xff0…

【毕业设计】基于单片机的智能饮水控制系统 - 物联网 嵌入式 stm32 c51

文章目录 1 简介2 功能要求3 相关器件4 实现效果5 部分实现代码6 最后 1 简介 Hi&#xff0c;大家好&#xff0c;这里是丹成学长&#xff0c;今天向大家介绍一个学长做的单片机项目&#xff0c;不过这个项目是帮助企业研发的 基于单片机的智能饮水控制系统 大家可用于 毕业设…

对嵌入式开发方向的一些思考:在物联网方向

学习知识就是为了应用&#xff0c;我们学习了嵌入式硬件和软件、具备一定的基础之后&#xff0c;是不是很想找一些简单的项目练练手&#xff1f;如果你有这种非常强烈的感觉的话&#xff0c;我相信你已经具备一定的嵌入式开发技术基础了。 接下来&#xff0c;你可能要犯愁了&am…

毕业设计 基于stm32的居民小区火灾检测报警系统 - 物联网 嵌入式

基于stm32的居民小区火灾检测报警系统 由 STM32F103C8T6单片机最小系DHT11LCD1602显示模块蜂鸣器MQ-2ESP8266噪声模块按键3个。 系统内可以通过1602来显示温湿度值和烟雾浓度和噪声值和它们的阈值&#xff0c;可以通过按键设置阈值&#xff0c;当任意一个实际值大于阈值时蜂鸣…

物联网嵌入式系统开发应用软件公司怎么选择

软件开发已经不是新鲜词汇&#xff0c;而成为当下智能科技新潮流&#xff0c;对于不同的企业&#xff0c;嵌入式软件开发会有更针对性的需求&#xff0c;纵观整个软件开发行业&#xff0c;做APP小程序网站开发的多如牛毛&#xff0c;而物联网应用嵌入式技术开发公司并不多&…

聊一聊物联网嵌入式芯片的内容结构

摘要&#xff1a;物联网嵌入式芯片&#xff0c;存储是数据核心之一&#xff0c;这个对于编程和设计的人来说又比较陌生&#xff0c;今天来说说MCU中的内存结构。 本文分享自华为云社区《漫谈嵌入式系统的内存》&#xff0c;作者&#xff1a;o0龙龙0o 。 前言 物联网嵌入式芯片…

5G+边缘计算 物联网嵌入式边缘计算平台

伴随着5G的来临&#xff0c;物联网技术的发展趋势让互连机器设备和信息量猛增&#xff0c;调研机构Gartner公司预测&#xff0c;到2025年&#xff0c;75%的企业生成的数据将在边缘计算设施而不是在传统数据中心创建和处理。 为满足市场对高性能边缘计算平台的应用需求&#xff…

嵌入式物联网软件开发实战

一、使用STM32CubeMx 创建物联网项目软件工程 二、STM32CubeMx 配置时钟 三、STM32CubeMx 配置GPIO 及 GPIO 相关接口在软件工程中的形式及使用 四、STM32CubeMx 配置USART 串口 五、USART 串口在物联网项目中的使用之不定长数据接收&#xff08;DMA空闲中断方式&#xff09; 六…

视频教程-物联网嵌入式技术应用-物联网技术

物联网嵌入式技术应用 10年计算机网络行业经验&#xff0c;精通计算机网络;孰悉C、C、Java、C#等语言&#xff0c;熟悉tcp/IP协议结构;精通华为产品&#xff0c;思科产品,熟悉Linux操作平台&#xff0c;讲解条理清晰&#xff0c;对知识有自己独到见解&#xff0c;善于发散学生的…

物联网嵌入式STM32资料大全,超100G

物联网嵌入式 STM32资料大全&#xff0c;超100G 本文是提供物联网&#xff0c;嵌入式&#xff0c; stm32等资料大全的&#xff0c;都是各类资料大全&#xff0c;下面图片只是网盘链接的截图&#xff0c;已经是有生之年系列了&#xff0c;别要求那么高了&#xff0c;自己到…

物联网嵌入式开发人员面临的5大挑战

​  开发嵌入式软件并不像以前那样“简单”。对于许多开发团队来说&#xff0c;创建一个独立的设备仍然经常是具有挑战性的。在物联网时代&#xff0c;我们开始将一切都连接起来&#xff0c;系统的复杂性正在飙升&#xff0c;传统的嵌入式开发人员发现&#xff0c;他们面临着…

嵌入式与物联网的特点分析,它们有什么关系?

虽然嵌入式系统已经有30多年的历史&#xff0c;但是原来一直隐藏在背后的&#xff0c;自从物联网上升为国家战略后&#xff0c;嵌入式系统也从后台走到前台。本文主要介绍的是嵌入式与物联网之间的关系&#xff0c;其次介绍了嵌入式及物联网的特点及特征&#xff0c;具体的跟随…

嵌入式物联网系统软硬件基础知识大全

关注、星标公众号&#xff0c;直达精彩内容 来源&#xff1a;网络素材 本文主要介绍嵌入式系统的基础知识&#xff0c;涉及嵌入式软件和硬件的方方面面&#xff0c;希望对各位有帮助。 嵌入式系统基础 1、嵌入式系统的定义 &#xff08;1&#xff09;定义&#xff1a;以应用为中…

嵌入式系统与物联网的关系

嵌入式系统与物联网的关系 一、物联网与嵌入式的基本概念物联网定义定义 1定义 2定义 3定义 4 嵌入式定义IEEE 的定义 嵌入式与物联网的区别和联系 二、从构成模型看物联网与嵌入式三、物联网时代嵌入式系统的华丽转身从计算机的历史说起嵌入式应用环境物联应用本质 从单片机到…

PHP反序列化笔记

目录 文章目录 目录private变量与protected变量序列化后的特点序列化后的字段长度前面可以加 题目解题步骤 CVE-2016-7124漏洞介绍演示代码题目解题步骤 PHP Session 反序列化PHP的3种序列化处理器安全问题当 session.auto_start&#xff1d;Off 时测试Demo 题目解题步骤 phar反…

2022/3/28 PHP反序列化

小迪 参考&#xff1a; https://www.bilibili.com/video/BV1JZ4y1c7ro?p36&spm_id_from333.880.my_history.page.click https://blog.csdn.net/Hardworking666/article/details/122373938 https://blog.csdn.net/Hardworking666/article/details/112725423 补充&#xff…

PHP反序列化漏洞-从入门到提升

目录 第一章 PHP序列化基础 1.1 PHP序列化 1.1.1 PHP序列化概述 1.1.2 标准序列化 1.1.3 自定义序列化 1.1.4 序列化存储和转发 1.2 PHP反序列化 1.2.1 标准反序列化 1.2.2 未定义类的反序列化 1.2.3 Protected、Private属性反序列化 1.3 PHP序列化相关magic函数 1…

php反序列化及__toString()

思路&#xff1a; 1./?sdata://text/plain,XXXXXX 2.include …php 3._tostring() echo unserialize(pass) echo file_get_content($this-file&#xff09; 把反序列化属性的值读取并输出 主要两个文件 index.php <?php $user $_GET["user"]; $file $_GET[…

PHP反序列化字符串逃逸

例题可看&#xff1a;https://www.cnblogs.com/v2ish1yan/articles/16118319.html 今天才学的&#xff0c;做个记录。 字符串逃逸分为两种&#xff0c;减少和增多。 主要是通过一个preg_replace()函数来进行字符串的减少和增多。 首先要知道逃逸的原理&#xff0c;就是反序…