电赛练习之旋转倒立摆

电赛练习之旋转倒立摆2019年电赛已经结束,虽然结果不能令人满意,但闲下来,还是总结一下电赛学到的东西与失败的地方。这一次先来谈一下一阶旋转倒立摆。一、题目分析:拿到一道题目,其实最应该做的事情是分析题目,因为我们往往可以发现某些发挥题是在基础题的基础上进行的,但是,可能某些发挥题需要在基础题的基础上修改结构,我们也可以发现,题目中的某些问题具有相似性,当我们合并同类项的时候,可以把题目的要求变得简单。一下,我粘…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE使用 1年只要46元 售后保障 童叟无欺

2019年电赛已经结束,虽然结果不能令人满意,但闲下来,还是总结一下电赛学到的东西与失败的地方。这一次先来谈一下一阶旋转倒立摆。

一、题目分析:

拿到一道题目,其实最应该做的事情是分析题目,因为我们往往可以发现某些发挥题是在基础题的基础上进行的,但是,可能某些发挥题需要在基础题的基础上修改结构,我们也可以发现,题目中的某些问题具有相似性,当我们合并同类项的时候,可以把题目的要求变得简单。一下,我粘贴过来2013年一阶旋转倒立摆的题目以及要求:
在这里插入图片描述
在这里插入图片描述在这里插入图片描述
我们很容易知道本题的核心是使得摆杆保持稳定,之后再其基础上进行一些功能的延伸。为了更好地理解题意,我做了如下的分析:

一阶旋转到倒立摆

手动起摆

自动起摆

摆杆的运动

电动机的控制

可以看到,题目中所要求完成的就是通过对一个电机的控制来实现的,控制题的核心还是要落在控制上。历届电子设计大赛经常被人戏称为电子机械设计大赛,虽然控制题的核心是落在控制上,但系统的机械结构也至关重要。此次练习采用的是平衡小车之家的套件,所以省去了搭硬件的很多麻烦。单单就完成题目要求的功能来说,数值分析并不是很必要(强迫症或者冲击国奖者除外)。因此,本篇内容仅仅从代码的角度来进行分析。
俗话说的好,不写注释的程序猿是流氓。我个人很喜欢在主函数的前面写上所用引脚的定义,这样一来方便接线,二来可以避免引脚重复使用,造成不必要的麻烦。

/************************************************ 圆周倒立摆实验 ADC:PA1 Motor:PC0 PC1 Pwm:PB4 PB5 编码器A相:PB6 B相:PB7 按键:PA0 2 3 4 @JackFu ************************************************/

先来谈一谈我程序的风格,
一、
在主函数里我将初始化的函数写在了Board_Init();这个函数里,这样看起来比较方便后期的更改与检查,减少了主函数的代码量,使得主函数看起来简洁、调理。

void Board_Init(void)
{ 
   
	delay_init();
	Adc_Init();
	KEY_Init();
	Ecoder_TIM4_Init(); 
	Motor_Init();
	Timer_pwm_Init(999,8);
// Timer_Init_TIM5(4999,71);
// TIM_SetCompare2(TIM3,800);
	uart_init(115200);
	NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
}

二、我一般习惯于建立两个文件夹:HARDWARE、CONTROL。其中前者用来存放于硬件相关的一些配置函数,后者用来放与软件相关的一些配置函数。这样方便后期的维护与移植。

对于其中用到的基本配置,如ADC,ENCODER,MOTOR等等,此处不再赘述。网上有很多的例程与讲解。
该题的核心控制为电机,关键函数如下:

//电机调节函数 (控制电机倒立,很重要)
void Motor_Config(short pwm)
{ 
   
	int speed;
	if(start == 1)                         //开启倒立,调节倒立摆平衡
	{ 
   
		if(pwm<0 && pwm > -PWM_MAX)
		{ 
   
			speed = PWM_MAX + pwm;
			if(speed < 0)
			{ 
   
				speed = 0;
			}
			TIM_SetCompare2(TIM3,speed);
			Motor_reverse();
		}
		else if(pwm > 0 && pwm < PWM_MAX)
		{ 
   
			speed = PWM_MAX - pwm;
			if(speed > 400)
			{ 
   
				speed = 400;
			}
			TIM_SetCompare2(TIM3,speed);
			Motor_Forward();
		}
	}
	else                                   //电位器偏角过大,无法调节,为了安全,关闭电机
	{ 
   
		Motor_stop();
	}
}

传入的参数为电机ENCODER,电位器ADC采集值后经过PID处理的数据,具体的处理函数如下:

short PID_Pos_PosCalc(short NextPoint)
{ 
   
  register float  iError,dError;
	iError = pos_Set_Pos - NextPoint;        				// 偏差
	pos_SumError += iError*0.2;				    				// 积分
	if(pos_SumError > 1000.0)								//积分限幅1000
		pos_SumError = 1000.0;
	if(pos_SumError < -1000.0)
		pos_SumError = -1000.0;	
	dError = iError - pos_LastError; 						// 当前微分
	pos_LastError = iError;
	
	return(short)(pos_p * iError*0.8 + pos_i * pos_SumError + pos_d * dError);	//返回计算值
}


short PID_Ang_PosCalc(short NextPoint)
{ 
   
  register float  iError,dError;
	iError = ang_Set_Pos - NextPoint;        				// 偏差
	ang_SumError += iError;				    				// 积分
	if(ang_SumError > 1000.0)								//积分限幅1000
		ang_SumError = 1000.0;
	if(ang_SumError < -1000.0)
		ang_SumError = -1000.0;	
	dError = iError - ang_LastError; 						// 当前微分
	ang_LastError = iError;
	
	return(short)(ang_p * iError + ang_i * ang_SumError + ang_d * dError);	//返回计算值
}
void Task3(void)
{ 
   
	short temp,temp1,temp2;
	temp = 0;
	temp1 = ANG_MIN_VAL;										//角度最大值
	temp2 = ANG_MAX_VAL;										//角度最小值
	
	ang_Cur_Pos = Get_Adc_Average(ADC_Channel_1,15);			//获取当前角度
	
	
	ang_Set_Pos = ANG_MID_VAL;									//将平衡时位置角度应当采样的值赋给设定值
	
	if((ang_Cur_Pos > temp1)&&(ang_Cur_Pos<temp2))
	{ 
   
		temp += PID_Ang_PosCalc(ang_Cur_Pos);					//角度还在可以调整的范围内 
		start = 1;		
	}
	
	else
	{ 
   															//角度过大,调整不了,为了安全,关机
		start = 0;
	}
	
// if(++Position_Target>5)
		pos_Cur_Pos = PID_Pos_PosCalc(Encoder),Position_Target=0;
	temp -= pos_Cur_Pos;
// if(pos_Cur_Pos > 0 && flag == -1)
// { 
   
// ANG_MID_VAL = ANG_MID_VAL-5;
// flag = flag*(-1);
// }
// else if(pos_Cur_Pos < 0 && flag == 1)
// { 
   
// ANG_MID_VAL = ANG_MID_VAL+5;
// flag = flag*(-1);
// }

	Motor_Config(temp);
	printf("%d ",temp);
	printf("%d\r\n",Encoder);
}

该题的控制采用了双环的PID来进行处理,在进行学习的过程中发现,仅仅采用电位器环的PID可以使得摆杆倒立起来,但整个摆杆会整体顺时针或者逆时针的旋转,并不能满足题目要求。正如前面所述,在考虑机械结构时,要整体考虑,否则当做到这里时,发现需要用双环来进行控制,再改机械结构,会浪费很多的时间。

这里附上源码 (链接: Mycode(双环控制)

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/171008.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • DOS命令 tasklist

    DOS命令 tasklist在WindowsXP中使用“Ctrl+Alt+Del”组合键,进入“Windows任务管理器”,在“进程”选项卡中可以查看本机完整的进程列表,而且可以通过手工定制进程列表的方式获得更多的进程信息,如会话ID、用户名等,但遗憾的是,我们查看不到这些进程到底提供了哪些系统服务。其实,在WindowsXP中新增的一个命令行工具“Tasklist.exe”就能实现上面的功能。作用:用来显示运行在本地

    2022年5月29日
    30
  • JVM内存模型详解「建议收藏」

    JVM内存模型详解「建议收藏」笔记大纲1、jvm内存结构图2、jvm按照线程共享和私有内存区域划分结构图3、堆和栈在功能、内存大小、线程共享私有进行比较4、JVM运行结构图5、线程安全本质时序图6、jdk6、7、8三个版本内存模型比较7、jdk1.8为什么将方法区移除到本地内存8、jvm内存启动参数详解JVM内存结构图(JDK1.6)多线程共享内存区域:方法区、堆。每一个线程独享内存:java栈、本地方法栈、程序计数器。程序计…

    2022年6月4日
    32
  • vue开发移动端app苹果手机的bug脱坑

    vue开发移动端app苹果手机的bug脱坑vue 开发移动端 app 网页打包苹果 app 的坑总结列表设置 overflow auto 后 滚动效果不流畅的问题 可以在列表设置 flex 1 overflow y auto webkit overflow scrolling touch 在 iOS 中出现滚动卡顿问题上诉解决方法还会出现一个问题 就是会导致在列表中如果有弹窗 position fixed 会导致弹窗被覆盖或者覆盖不完全的问题 为此需要将弹窗放在列表外 下面为代码例子

    2025年10月30日
    4
  • 使用ctk库

    使用ctk库上篇文章写的如何生成一个简易ctk动态库https://blog.csdn.net/qq_16238157/article/details/86602476这篇文章写如何简易的使用交代路径下图上篇文章已经写过关于ctk动态库如何生成下面介绍一下各个文件夹:CTK文件夹:是ctk的源码ctkWork文件夹:用vs编译生成的ctk插件myCTK文件夹:是网上找的一个调用ct…

    2022年6月3日
    70
  • 什么是itp_常见的培训类型

    什么是itp_常见的培训类型PMP培训知识体系中,项目管理信息系统PMIS是个经常出现的工具和技术,它即属于事业环境因素,会制定我们项目的规划和其它工作,同时也是工具和技术,我们需要依靠和借助PMIS完成很多工作的实际操作,很多

    2022年8月5日
    7
  • XSD文件结构详解「建议收藏」

    XSD文件结构详解「建议收藏」XSD(xmlSchemaDefinition)XmlSchema的用途1.  定义一个Xml文档中都有什么元素2.  定义一个Xml文档中都会有什么属性3.  定义某个节点的都有什么样的子节点,可以有多少个子节点,子节点出现的顺序4.  定义元素或者属性的数据类型5.  定义元素或者属性的默认值或者固定值XmlSchema的根元素:&lt;?xmlversion="1.0"?&gt;…

    2025年7月25日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号