OpenEdv-开源电子网

 找回密码
 立即注册
正点原子全套STM32/Linux/FPGA开发资料,上千讲STM32视频教程免费下载...
查看: 11230|回复: 11

ADC采集数据,冒泡排序寻找最大值

[复制链接]

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
发表于 2015-6-14 16:03:32 | 显示全部楼层 |阅读模式
5金钱


思路:adc的DMA采集数据,将采集回来的数据存入数组中,通过冒泡排序的,找出最大值,将最大值显示在液晶上,不加这个冒泡排序的算法数据是正确的,可是一加上这个,怎么数据就成了0-1、20的数据了。这不科学啊。Mode是按键进行通道切换的,这样写的有问题吗?

 

 

[mw_shl_code=c,true]/**************************************************************************** * 名 称:ADCValueDisplay_MainLoop(void) * 功 能:ADC采集数据输出函数,液晶显示函数 * LCD显示电压值主循环 * 入口参数: * 出口参数:无 * 说 明: * 调用方法:直接调用 ****************************************************************************/ void ADCValueDisplay_MainLoop(u8 Mode_flag) { static u16 Mode; static float uwVoltage =0, uwMVoltage=0; static u32 i,j,k,value1=0,value2=0,temp1,temp2; Mode=Mode_flag; //把ADC采样值转换成电压值 if(1==Mode) { // GUI_DispDecAt(uwADC3ConvertedVoltage*3300/0xFFF,210,90,4); for(i=0;i<N;i++) { ArrDataBuffer =uwADC3ConvertedVoltage*3300/0xFFF; } //采样由小到大排列,排序采样冒泡法 for(j=0;j<N-1;j++) //冒泡 { for(k=0;k<N-1-j;k++) { if(ArrDataBuffer>ArrDataBuffer[i+1]) { temp1=ArrDataBuffer; ArrDataBuffer=ArrDataBuffer[i+1]; ArrDataBuffer[i+1]=temp1; } } } value1=(ArrDataBuffer[i+1]+ArrDataBuffer+ArrDataBuffer[i-1])/3;//3次平均 // printf("uwADC3ConvertedVoltage is: %d\r\n",uwADC3ConvertedVoltage ); //uwVoltage = uwADC3ConvertedVoltage/1000.0; GUI_SetFont(&GUI_Font24_1); GUI_SetColor(GUI_YELLOW); GUI_DispDecAt(value1,120,90,4); //实际电压值 // printf("PA3: %0.2f V\r\n",uwVoltage); } if(2==Mode) { // GUI_DispDecAt(uhADC1ConvertedValue*3300/0xFFF,210,120,4);// mV for(i=0;i<N;i++) { ArrDataBuffer1 =uhADC1ConvertedValue*3300/0xFFF; } //采样由小到大排列,排序采样冒泡法 for(j=0;j<N-1;j++) //冒泡 { for(k=0;k<N-1-j;k++) { if(ArrDataBuffer1>ArrDataBuffer1[i+1]) { temp2=ArrDataBuffer1; ArrDataBuffer1=ArrDataBuffer1[i+1]; ArrDataBuffer1[i+1]=temp2; } } } value2=((ArrDataBuffer1[i+1]+ArrDataBuffer1+ArrDataBuffer1[i-1])*2)/3; //3次平均 // printf("uwADC1ConvertedVoltage is: %d\r\n",uwADC1ConvertedVoltage ); // uwMVoltage = uwADC1ConvertedVoltage/1000.0; GUI_SetFont(&GUI_Font24_1); GUI_SetColor(GUI_YELLOW); GUI_DispDecAt(value2,120,120,4); //实际电压值 // printf("PA6: %0.2f V\r\n",uwMVoltage); } }[/mw_shl_code]


 

正点原子逻辑分析仪DL16劲爆上市
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-14 16:35:17 | 显示全部楼层
采用ADC DMA软件的触发的方式
回复

使用道具 举报

170

主题

917

帖子

0

精华

金牌会员

Rank: 6Rank: 6

积分
1824
金钱
1824
注册时间
2013-4-13
在线时间
63 小时
发表于 2015-6-14 17:27:39 | 显示全部楼层
1,可能你的冒泡排序函数有问题,建议仿真一下;2,采集的值是瞬息万变的,显示出来会有振动;3,思路换一下
手艺人要内外兼修。
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-14 23:42:39 | 显示全部楼层
回复【2楼】ahsssxd:
---------------------------------
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-14 23:43:20 | 显示全部楼层
貌似是的,我明天再看看
回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
7952
金钱
7952
注册时间
2014-8-13
在线时间
1590 小时
发表于 2015-6-15 19:05:57 | 显示全部楼层
如果你只显示最大值,为什么要全部冒泡?  你只选出最大N个数值,做平均不就可以了?

而且为什么要从AD值转换成电压值之后再冒泡?  先从AD值中选出最大N个数值,求平均,然后在换算成电压(仅一次换算)。

冒泡循环貌似没什么问题,内循环每次都把循环内的数据最大的移动到最右边,外循环每次都使得内循环少移动一位。好像很有道理。如果有问题,自己下断点调试下看看吧。懒得帮你看了
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-16 09:53:46 | 显示全部楼层
回复【6楼】mack13013:
------------------------
我主要是来找最大值的哦,其他的数据对我来说用处不大
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-16 09:55:33 | 显示全部楼层

[mw_shl_code=c,true] [/mw_shl_code] [mw_shl_code=c,true] /**************************************************************************** * 名 称:ADCValueDisplay_MainLoop(void) * 功 能:ADC采集数据输出函数,液晶显示函数 * LCD显示电压值主循环 * 入口参数: * 出口参数:无 * 说 明: * 调用方法:直接调用 ****************************************************************************/ void ADCValueDisplay_MainLoop(u8 Mode_flag) { static u16 Mode; static u32 i,j,k,value,temp; static u32 ArrDataBuffer[N];//N个数据 Mode=Mode_flag; if(1==Mode) { value=GlideaverageValueFilter(uwADC3ConvertedVoltage*3300/0xFFF); // printf("uwADC3ConvertedVoltage is: %d\r\n",uwADC3ConvertedVoltage ); GUI_SetFont(&GUI_Font24_1); GUI_SetColor(GUI_RED); GUI_DispStringAt("V",260,90); GUI_SetColor(GUI_YELLOW); GUI_DispDecAt(value,120,90,4); GUI_DispStringAt(".",220,90); GUI_DispDecAt(value/1000,210,90,1); GUI_DispDecAt((value%1000)/10,230,90,2); //实际电压值 // printf("PA3: %0.2f V\r\n",uwVoltage); } if(2==Mode) { value=GlideaverageValueFilter(uwADC1ConvertedVoltage*3300/0xFFF)*2; // printf("uwADC1ConvertedVoltage is: %d\r\n",uwADC1ConvertedVoltage ); GUI_SetFont(&GUI_Font24_1); GUI_SetColor(GUI_RED); GUI_DispStringAt("V",260,120); GUI_SetColor(GUI_YELLOW); GUI_DispDecAt(value,120,120,4); GUI_DispStringAt(".",220,120); GUI_DispDecAt(value/1000,210,120,1); GUI_DispDecAt((value%1000)/10,230,120,2); //实际电压值 // printf("PA6: %0.2f V\r\n",uwMVoltage); } } /**************************************************************************** * 名 称:?void ADC_Key(void) * 功 能: 按键处理函数,设置ADC采集通道 * 入口参数: * 出口参数:无 * 说 明: * 调用方法:直接调用 ****************************************************************************/ void ADC_Key(void) { key=KEY_Scan(0); //得到键值 if(key) { switch(key) { case KEY1_PRES: ADC_flag=1; break; case KEY2_PRES: ADC_flag=2; break; } } else delay_nms(10); } /**************************************************************************** * 名 称:?char GlideaverageValueFilter(char Value) * 功 能: 递推平均滤波 * 入口参数: * 出口参数: * 说 明:带返回值的函数 * 调用方法:直接调用 ****************************************************************************/ u32 GlideaverageValueFilter(u32 Value) { static u32 i,value; static u32 Sum; Sum=0; Data[10]=Filter(Value); //采集数据放在最高位 mV for(i=0;i<10;i++) { Data=Data[i+1]; //所有数据左移,地位扔掉 Sum+=Data; } value=Sum/10; return(value); } /**************************************************************************** * 名 称:?uchar MaxValueFilter(u32 Value) * 功 能: 寻找最大值 * 入口参数: * 出口参数: * 说 明:带返回值的函数 * 调用方法:直接调用 ****************************************************************************/ u32 MaxValueFilter(u32 Value) { static u32 i,j,k,value,temp; static u32 ArrDataBuffer[N];//2400个数据 for(i=0;i<N;i++) { ArrDataBuffer = GlideaverageValueFilter(Value); } //采样由小到大排列,排序采样冒泡法 for(j=0;j<N-1;j++) //冒泡 { for(k=0;k<N-1-j;k++) { if(ArrDataBuffer[k]>ArrDataBuffer[k+1]) { temp=ArrDataBuffer[k]; ArrDataBuffer[k]=ArrDataBuffer[k+1]; ArrDataBuffer[k+1]=temp; } } } value=ArrDataBuffer[N-1]; return(value); }[/mw_shl_code]

回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
7952
金钱
7952
注册时间
2014-8-13
在线时间
1590 小时
发表于 2015-6-16 10:03:28 | 显示全部楼层
回复【7楼】ahsssxd:
---------------------------------
你回帖不看贴的。

你冒泡外循环改为for(j=0;j<3;j++) 就可以求出最大的3个值了。

另外,我5楼回帖中的N跟你程序里的N不是一个值,我5楼回帖中的N是一个若干值,比如上一行中的3。为了便于你理解,你可以将我5楼回帖中的N改成m。
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-16 10:30:54 | 显示全部楼层
回复【9楼】mack13013:
---------------------------------
你的建议我再试试,现在出现的问题比较多,采集回来2400个点,还没有找到最大值,采集的都为中间的数值
回复

使用道具 举报

33

主题

984

帖子

1

精华

论坛元老

Rank: 8Rank: 8

积分
7952
金钱
7952
注册时间
2014-8-13
在线时间
1590 小时
发表于 2015-6-16 10:45:06 | 显示全部楼层
回复【10楼】ahsssxd:
---------------------------------
如果AD采集精度有问题,你可以参考我的一个帖子,

分享:STM32F407的ADC真的不够精准吗?教你设置STM32F407的ADC参数使其精度高于千分之一
http://www.openedv.com/posts/list/51312.htm。

虽然帖子是针对STM32F407的,但是原理是共通的,因为在STM32F103上我也做过同样测试。

帖子虽然能解决采集的数值“抖动”范围比较大的问题,但是无法确定ADC的数值是否有“漂移”。
回复

使用道具 举报

11

主题

108

帖子

0

精华

中级会员

Rank: 3Rank: 3

积分
492
金钱
492
注册时间
2015-3-3
在线时间
114 小时
 楼主| 发表于 2015-6-16 11:11:34 | 显示全部楼层
回复【11楼】mack13013:
---------------------------------
我采集的是一个锯齿波信号,我来直接采集电平信号都是对的,值也就误差才十几之间
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则



关闭

原子哥极力推荐上一条 /2 下一条

正点原子公众号

QQ|手机版|OpenEdv-开源电子网 ( 粤ICP备12000418号-1 )

GMT+8, 2025-2-24 18:08

Powered by OpenEdv-开源电子网

© 2001-2030 OpenEdv-开源电子网

快速回复 返回顶部 返回列表