找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 45746|回复: 34
打印 上一主题 下一主题
收起左侧

GP2D12红外测距传感器原理与单片机源程序等详细资料

  [复制链接]
跳转到指定楼层
楼主
GP2D12红外测距传感器使用介绍
一、“看到”障碍物的眼睛——红外测距传感器
     机器视觉中最接近人眼的莫过于摄像头了,可图像处理小车的“大脑”对付不了,至少目前的“大脑”能力不够,等进化后也许能够应付。  

     为了能“看到”障碍物,小车目前能用的主要是各类测距传感器,典型的有超声波和红外两种,此外还有利用光线的反射强弱来判断的,这种方式不具备“测距”功能,但可以判断有无!因为不同物体表面及颜色反射的能力不同(看后面的数据)。  
     本文主要讨论的是机器人中最常用的红外测距传感器  ——  GP2D12。  

     首先认识一下:  


模块共三个接口:红线---VCC_5V;黑线---GND;黄线—PC5(ADC数据采样)
将模块通电,黄线接到ADC通道输入端即可工作;将采样得到的电压值通过填入表格,进行线性化处理,得到线性化公式。通过公式,可将ADC采样值转化为距离值。实测,在10cm—30cm范围内,较为准确,最大误差在1cm。模块对被测角度的灵敏度很高,同一位置,不同的角度,误差可以超过5cm,最好的测量角度是90度。

   之所以选择 GP2D12  红外测距传感器,理由如下:  
首先是因为在机器人活动中(不包含工业机器人)这个传感器最常用,几乎每家国外的机器人配件供应商都提供。使用英文版搜索一下 “MiniSumo”,你将会发现 GP2D12使用是多么普遍。   

    其二是因为它的测距范围和小车的“个头”及运动速度匹配,对于 10cm 见方、运动速度10 – 30cm/s  的小个头,能“看到”几米开外的东西意义不大,而 10 – 80cm  正是它所要关注的范围。  
    GP2D12的工作原理我理解为(仅供参考,欢迎指正):  
    它是由一个红外发射管和一个 PSD(Position Sensing Device位置敏感检测装置)以及相应的计算电路构成,Sharp  公司的 PSD 很有特色,它可以检测到光点落在它上面的微小位移, 分辨率达微米,GP2D12 正是利用这个特性实现了几何方式测距。  

    红外发射管发出的光束,遇到障碍物反射回来,落在 PSD 上,构成了一个等腰三角形, 借助于 PSD 可以测得三角形的底,而两个底角是固定的,由发射管确定,此时便可通过底边 推算出高,也就是我们所要的距离。如下图所示:  

测量原理示意图

  从图中可以看出,这是一个顶角极锐的等腰三角形,底边只有2cm  ,高却要有10 – 80cm,所以 PSD 的分辨率必须极高,否则微小的偏差都会带来距离的巨大误差。从这一点也可以得出,它的测距结果很难稳定、精确,毕竟比值太大。  

     因为 PSD 的尺寸有限,从图中就很容易理解为何它的测量距离超出范围后就不可能是有效数据,连趋势都得不到。  

     从上述原理描述还可以知道,它不是连续测量,得到底边长度后,必须经过计算才能得到距离值,然后转换为模拟信号输出。  

    这两个推论在那篇“Sharp  GP2D12  applicationNote”(应用指南)有所印证,具体表现为它测距的强指向性和输出的不确定性(噪音高达  200mV,相对于 2.4V 的满量程输出而言达5%)。 这篇文章好像是国外一个爱好者写的,他做了大量的测试,对使用者掌握GP2D12 的性能及合理的使用它极有帮助。  



    总有人问     GP2D12  是否能用于某些场合?如果能仔细吃透上述指标,自然会有答案。 还有人问它与超声波传感器那个好,我想这些指标也会告诉你!  至于更详细的内容,读者可进一步阅读GP2D12的数据手册以及上面介绍的应用指南, 在此我就不再赘述。  

   在硬件上,没有太多的难度,但是要用好GP2D12,软件上似乎要做些努力,必须解决的有两个问题:  

    一是信号的线性化 ,因为输出与距离的关系是非线性的,为便于程序中使用距离信息,必须将模拟信号转换为相应得距离值。  
    二是滤波 ,因为按照上述应用指南的测量分析,GP2D12的输出噪声很大;此外,还由于测量的非连续性,导致连续的距离变化对应的输出为阶跃信号,也需要通过滤波将其平滑。  

2.1  线性化  
    关于线性化,开始时我也一筹莫展,曾想用折线近似实现,但尝试后觉得代码量太大,而且需要做大量数据采集。
    后来在acroname网站上(二年前),发现了一个极好的“东东”  —个用 Excel 制作的电子表,表格的格式如下:

gp2d12-使用表格(本文附件可下载)

    里面有作者根据 GP2D12 特性建立的数学模型(线性化公式),并预留的使用者输入参数的地方,只需按其要求填入:  
    AD 的位数、AD 供电电压(满量程),并采集 8 点(10cm 间隔)GP2D12  的输出电压, 填入表中,它就可自动生成线性化公式的参数 ,提供了整形和浮点两种格式,还附有由此产生的结果与实际的偏差表,并用生动的图形表示,十分直观、实用。  
    此表可在embedream的相关资料中下载,本该提供它的原始链接,无奈现在没有了,只找到了一个类似的文档  —  Sharp IR Range Finder Voltage-to-Range Conversion Article 内容也是讨论线性化的,读者不妨一读。配合此文也许更容易理解使用那张Excel 表格。


     根据上述公式及程序得到的结果如下:  
                             GP2D12 不同颜色测距结果对比  

    第一列为实际距离,第二列障碍物表面为白纸,第三列障碍物为褐色木盒,读者可比照Excel 表中的数据,可以看出基本吻合。同时还可以从上面数据中看出,GP2D12  确实如其手册中所说,基本不受障碍物的颜色影响。  



2.2  滤波
    滤波主要解决两个问题,一是在GP2D12恒定输出阶段,按应用指南的分析,有不小的噪声,需要通过滤波消除。  
    二是由于其非连续测量的特性,导致其测量连续变化的距离时,输出是阶跃形式的,这对程序判断极为不利,为了弱化这个影响,也期望通过滤波实现。  
    根据 GP2D12 的手册,其测量周期为  40ms  左右(38ms),综合小车单片机的内存及处理需求,采用 5ms 采样一次,取最近 8 次的结果平均值的滤波方式,也就是说,一个测量周期采 8 个数据平均。  
    这样处理可以降低噪声的影响,这点容易理解。至于弱化阶跃信号,不知读者是否认同?  
    我是这样考虑:在出现阶跃信号时,8 个数据中随着时间推移,新的信号所占的权重不断加大,使得信号逐渐从前一个信号平缓的过渡到新的信号上。但是这样处理,导致了距离信号反映滞后,要到下一个信号快到时,本次的输出才接近本次的信号。就这一点而言,似 乎有些不尽合理,有待读者深入探讨。  

红外测试数据:
10cm   11.55--11.66
15cm  16.15--16.38
16.5cm  17.46--17.92
18cm   20.01--20.13
19cm  19.44--19.81
20cm  20.01--20.15
22cm  21.56--21.82
23cm  23.20--23.66
24.5cm  23.96--24.23
28cm  27.39--27.93


stm32单片机红外测距源程序如下:
  1. /******************** (C) COPYRIGHT 2007 STMicroelectronics ********************
  2. * File Name          : main.c
  3. * Author             : MCD Application Team
  4. * Version            : V1.0
  5. * Date               : 10/08
  6. * Description        : Main program body
  7. ********************************************************************************
  8. * THE PRESENT SOFTWARE WHICH IS FOR GUIDANCE ONLY AIMS AT PROVIDING CUSTOMERS
  9. * WITH CODING INFORMATION REGARDING THEIR PRODUCTS IN ORDER FOR THEM TO SAVE TIME.
  10. * AS A RESULT, STMICROELECTRONICS SHALL NOT BE HELD LIABLE FOR ANY DIRECT,
  11. * INDIRECT OR CONSEQUENTIAL DAMAGES WITH RESPECT TO ANY CLAIMS ARISING FROM THE
  12. * CONTENT OF SUCH SOFTWARE AND/OR THE USE MADE BY CUSTOMERS OF THE CODING
  13. * INFORMATION CONTAINED HEREIN IN CONNECTION WITH THEIR PRODUCTS.
  14. *******************************************************************************/

  15. /* Includes ------------------------------------------------------------------*/
  16. #include "stm32f10x_lib.h"
  17. #include "stdio.h"
  18. /* Private macro -------------------------------------------------------------*/
  19. #define countof(a)   (sizeof(a) / sizeof(*(a)))

  20. /* Private typedef -----------------------------------------------------------*/
  21. #define TxBufferSize   (countof(TxBuffer) - 1)

  22. /* Private define ------------------------------------------------------------*/
  23. u8 TxBuffer[] = "\n\rADC Example1: ADC TO DMA TO UART1\n\r";
  24. u8 TxCounter = 0;

  25. /* Private define ------------------------------------------------------------*/
  26. #define ADC1_DR_Address    ((u32)0x4001244C)

  27. /* Private variables ---------------------------------------------------------*/
  28. USART_InitTypeDef USART_InitStructure;
  29. ADC_InitTypeDef ADC_InitStructure;
  30. DMA_InitTypeDef DMA_InitStructure;
  31. vu16 ADC_ConvertedValue;
  32. ErrorStatus HSEStartUpStatus;
  33.    
  34. /* Private function prototypes -----------------------------------------------*/
  35. void RCC_Configuration(void);
  36. void GPIO_Configuration(void);
  37. void NVIC_Configuration(void);
  38.   
  39. /* Private functions ---------------------------------------------------------*/
  40. void Delay_us(unsigned short us)
  41. {
  42.     unsigned short i;
  43.     while(us--)
  44.     {
  45.             for(i=0;i<10;i++);
  46.     }
  47. }

  48. void Delay_ms(unsigned short ms)
  49. {
  50.     unsigned short i;
  51.         while(ms--)
  52.     {
  53.         for(i=0;i<10000;i++);
  54.     }
  55. }

  56. //发送数据
  57. int fputc(int ch, FILE *f){
  58. USART_SendData(USART1,(unsigned char)ch);//USART1可以换成USART2等
  59.         while(!(USART1->SR&USART_FLAG_TXE));
  60. return(ch);
  61. }
  62. //接收数据
  63. int GetKey(void){
  64. while(!(USART1->SR&USART_FLAG_RXNE));
  65.         return((int)(USART1->DR&0x1FF));
  66. }

  67. /*******************************************************************************
  68. * Function Name  : main
  69. * Description    : Main program
  70. * Input          : None
  71. * Output         : None
  72. * Return         : None
  73. *******************************************************************************/
  74. int main(void)
  75. {
  76. //unsigned char i=1;
  77. unsigned long Tmp_Dat,i=1;
  78.         float distance,sum=0;
  79. #ifdef DEBUG
  80.   debug();
  81. #endif

  82.   /* System clocks configuration ---------------------------------------------*/
  83.   RCC_Configuration();

  84.   /* NVIC configuration ------------------------------------------------------*/
  85.   NVIC_Configuration();

  86.   /* GPIO configuration ------------------------------------------------------*/
  87.   GPIO_Configuration();

  88.   /* USART1 configuration ------------------------------------------------------*/
  89.   /* USART1 configured as follow:
  90.         - BaudRate = 9600 baud  
  91.         - Word Length = 8 Bits
  92.         - Two Stop Bit
  93.         - Odd parity
  94.         - Hardware flow control disabled (RTS and CTS signals)
  95.         - Receive and transmit enabled
  96.         - USART Clock disabled
  97.         - USART CPOL: Clock is active low
  98.         - USART CPHA: Data is captured on the second edge
  99.         - USART LastBit: The clock pulse of the last data bit is not output to
  100.                          the SCLK pin
  101.   */
  102.   USART_InitStructure.USART_BaudRate = 9600;
  103.   USART_InitStructure.USART_WordLength = USART_WordLength_8b;
  104.   USART_InitStructure.USART_StopBits = USART_StopBits_1;
  105.   USART_InitStructure.USART_Parity = USART_Parity_No;
  106.   USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
  107.   USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
  108.   USART_InitStructure.USART_Clock = USART_Clock_Disable;
  109.   USART_InitStructure.USART_CPOL = USART_CPOL_Low;
  110.   USART_InitStructure.USART_CPHA = USART_CPHA_2Edge;
  111.   USART_InitStructure.USART_LastBit = USART_LastBit_Disable;

  112.   /* Configure the USART1 */
  113.   USART_Init(USART1, &USART_InitStructure);

  114. /* Enable the USART Transmoit interrupt: this interrupt is generated when the
  115.    USART1 transmit data register is empty */  
  116.   USART_ITConfig(USART1, USART_IT_TXE, ENABLE);

  117. /* Enable the USART Receive interrupt: this interrupt is generated when the
  118.    USART1 receive data register is not empty */
  119.   USART_ITConfig(USART1, USART_IT_RXNE, ENABLE);

  120.   /* Enable USART1 */
  121.   USART_Cmd(USART1, ENABLE);

  122.   /* DMA channel1 configuration ----------------------------------------------*/
  123.   DMA_DeInit(DMA_Channel1);
  124.   DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;
  125.   DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;
  126.   DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
  127.   DMA_InitStructure.DMA_BufferSize = 1;
  128.   DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
  129.   DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;
  130.   DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
  131.   DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
  132.   DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
  133.   DMA_InitStructure.DMA_Priority = DMA_Priority_High;
  134.   DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
  135.   DMA_Init(DMA_Channel1, &DMA_InitStructure);
  136.   
  137.   /* Enable DMA channel1 */
  138.   DMA_Cmd(DMA_Channel1, ENABLE);
  139.      
  140.   /* ADC1 configuration ------------------------------------------------------*/
  141.   ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
  142.   ADC_InitStructure.ADC_ScanConvMode = ENABLE;
  143.   ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
  144.   ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
  145.   ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
  146.   ADC_InitStructure.ADC_NbrOfChannel = 1;
  147.   ADC_Init(ADC1, &ADC_InitStructure);

  148.   /* ADC1 regular channel1 configuration */
  149.   ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 1, ADC_SampleTime_55Cycles5);

  150.   /* Enable ADC1 DMA */
  151.   ADC_DMACmd(ADC1, ENABLE);
  152.   
  153.   /* Enable ADC1 */
  154.   ADC_Cmd(ADC1, ENABLE);

  155.   /* Enable ADC1 reset calibaration register */   
  156.   ADC_ResetCalibration(ADC1);
  157.   /* Check the end of ADC1 reset calibration register */
  158.   while(ADC_GetResetCalibrationStatus(ADC1));

  159.   /* Start ADC1 calibaration */
  160.   ADC_StartCalibration(ADC1);
  161.   /* Check the end of ADC1 calibration */
  162.   while(ADC_GetCalibrationStatus(ADC1));
  163.      
  164.   /* Start ADC1 Software Conversion */
  165.   ADC_SoftwareStartConvCmd(ADC1, ENABLE);

  166.   for(i=0;i<TxBufferSize;i++)
  167.   {
  168.    /* Write one byte to the transmit data register */
  169.   USART_SendData(USART1, TxBuffer[TxCounter++]);
  170.   while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  171.   while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  172.   }

  173.   while (1)
  174.   {
  175.                 Delay_ms(1000);
  176.                
  177.                 Tmp_Dat =  ADC_ConvertedValue;
  178.                 distance = (1/(Tmp_Dat*0.0000228324+0.00140335))-4.0;

  179.                         printf("\n%ld\n",Tmp_Dat);
  180.                    printf("%.2f",distance);

  181.                
  182.                
  183.                 Tmp_Dat = Tmp_Dat*3300/0x0fff;

  184.                 TxBuffer[0] = Tmp_Dat/1000+'0';
  185.                 TxBuffer[1] = '.';
  186.                 TxBuffer[2] = (Tmp_Dat%1000)/100+'0';
  187.                 TxBuffer[3] = (Tmp_Dat%100)/10+'0';
  188.                 TxBuffer[4] = Tmp_Dat%10+'0';
  189.                 TxBuffer[5] = 'V';

  190.                 USART_SendData(USART1, '[');
  191.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  192.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  193.                 USART_SendData(USART1, TxBuffer[0]);
  194.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  195.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  196.                 USART_SendData(USART1, TxBuffer[1]);
  197.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  198.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  199.                 USART_SendData(USART1, TxBuffer[2]);
  200.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  201.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  202.                 USART_SendData(USART1, TxBuffer[3]);
  203.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  204.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  205.                 USART_SendData(USART1, TxBuffer[4]);
  206.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  207.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  208.                 USART_SendData(USART1, TxBuffer[5]);
  209.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  210.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成

  211.                 USART_SendData(USART1, ']');
  212.                 while (!(USART1->SR & USART_FLAG_TXE));   //等待缓冲区空
  213.                 while (!(USART1->SR & USART_FLAG_TC));   //等待发送完成
  214.                 printf("\n");
  215.   }
  216. }

  217. /*******************************************************************************
  218. * Function Name  : RCC_Configuration
  219. * Description    : Configures the different system clocks.
  220. * Input          : None
  221. * Output         : None
  222. * Return         : None
  223. *******************************************************************************/
  224. void RCC_Configuration(void)
  225. {
  226.   /* RCC system reset(for debug purpose) */
  227.   RCC_DeInit();

  228.   /* Enable HSE */
  229.   RCC_HSEConfig(RCC_HSE_ON);

  230.   /* Wait till HSE is ready */
  231.   HSEStartUpStatus = RCC_WaitForHSEStartUp();

  232.   if(HSEStartUpStatus == SUCCESS)
  233.   {
  234.     /* Enable Prefetch Buffer */
  235.     FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable);

  236.     /* Flash 2 wait state */
  237.     FLASH_SetLatency(FLASH_Latency_2);
  238.   
  239.     /* HCLK = SYSCLK */
  240.     RCC_HCLKConfig(RCC_SYSCLK_Div1);
  241.   
  242.     /* PCLK2 = HCLK */
  243.     RCC_PCLK2Config(RCC_HCLK_Div1);

  244.     /* PCLK1 = HCLK/2 */
  245.     RCC_PCLK1Config(RCC_HCLK_Div2);

  246.     /* ADCCLK = PCLK2/4 */
  247.     RCC_ADCCLKConfig(RCC_PCLK2_Div4);
  248.   
  249.     /* PLLCLK = 8MHz * 7 = 56 MHz */
  250.     RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_7);

  251.     /* Enable PLL */
  252.     RCC_PLLCmd(ENABLE);

  253.     /* Wait till PLL is ready */
  254.     while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET)
  255.     {
  256.     }

  257.     /* Select PLL as system clock source */
  258.     RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK);

  259.     /* Wait till PLL is used as system clock source */
  260.     while(RCC_GetSYSCLKSource() != 0x08)
  261.     {
  262.     }
  263.   }

  264. /* Enable peripheral clocks --------------------------------------------------*/

  265.   /* Enable DMA clock */
  266.   RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA, ENABLE);

  267.   /* Enable ADC1 and GPIOC clock */
  268.   RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC, ENABLE);

  269.   /* Enable GPIOA and USART1 clocks */
  270.   RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_USART1, ENABLE);
  271. }

  272. /*******************************************************************************
  273. * Function Name  : GPIO_Configuration
  274. * Description    : Configures the different GPIO ports.
  275. * Input          : None
  276. * Output         : None
  277. * Return         : None
  278. *******************************************************************************/
  279. void GPIO_Configuration(void)
  280. {
  281.   GPIO_InitTypeDef GPIO_InitStructure;

  282.   /* Configure PC.05 (ADC Channel15) as analog input -------------------------*/
  283.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5;
  284.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
  285.   GPIO_Init(GPIOC, &GPIO_InitStructure);

  286.   /* Configure USART1 Tx (PA.09) as alternate function push-pull */
  287.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
  288.   GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  289.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
  290.   GPIO_Init(GPIOA, &GPIO_InitStructure);

  291.   /* Configure USART1 Rx (PA.10) as input floating */
  292.   GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
  293.   GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
  294.   GPIO_Init(GPIOA, &GPIO_InitStructure);
  295. }

  296. /*******************************************************************************
  297. * Function Name  : NVIC_Configuration
  298. * Description    : Configures Vector Table base location.
  299. * Input          : None
  300. * Output         : None
  301. * Return         : None
  302. *******************************************************************************/
  303. void NVIC_Configuration(void)
  304. ……………………

  305. …………限于本文篇幅 余下代码请从51黑下载附件…………
复制代码

所有资料51hei提供下载:

红外模块.zip (2.25 MB, 下载次数: 605)


评分

参与人数 2黑币 +71 收起 理由
sad520ljh + 21 绝世好帖!
admin + 50 共享资料的黑币奖励!

查看全部评分

分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏12 分享淘帖 顶4 踩
回复

使用道具 举报

沙发
ID:1 发表于 2017-11-25 22:51 | 只看该作者
好资料,51黑有你更精彩!!!
回复

使用道具 举报

板凳
ID:281489 发表于 2018-2-1 16:21 | 只看该作者
滤波方法学习了
回复

使用道具 举报

地板
ID:285576 发表于 2018-4-10 19:45 | 只看该作者
谢谢正需要。
回复

使用道具 举报

5#
ID:321214 发表于 2018-5-3 14:22 | 只看该作者
学到了
回复

使用道具 举报

6#
ID:344165 发表于 2018-6-3 19:31 | 只看该作者
谢谢学到了
回复

使用道具 举报

7#
ID:163643 发表于 2018-6-21 09:04 | 只看该作者
谢谢正需要。
回复

使用道具 举报

8#
ID:375755 发表于 2018-7-20 12:50 | 只看该作者
收下了
回复

使用道具 举报

9#
ID:36133 发表于 2018-9-25 20:58 | 只看该作者
学习一下
回复

使用道具 举报

10#
ID:422260 发表于 2018-11-7 12:11 | 只看该作者
非常想要
回复

使用道具 举报

11#
ID:434718 发表于 2018-11-28 15:20 | 只看该作者
51有你更精彩
回复

使用道具 举报

12#
ID:93667 发表于 2018-12-4 15:00 | 只看该作者
谢谢。。。。。。学习了
回复

使用道具 举报

13#
ID:287515 发表于 2019-1-19 23:57 | 只看该作者
谢谢分享!
回复

使用道具 举报

14#
ID:483145 发表于 2019-3-2 11:33 | 只看该作者
资料好详细
回复

使用道具 举报

15#
ID:524497 发表于 2019-4-29 15:48 | 只看该作者
学到了,谢谢
回复

使用道具 举报

16#
ID:223705 发表于 2019-6-4 08:49 | 只看该作者
学习了,正在玩红外测距!
回复

使用道具 举报

17#
ID:389153 发表于 2019-7-12 17:29 | 只看该作者
好资料 值得学习 多谢分享
回复

使用道具 举报

18#
ID:389153 发表于 2019-7-12 17:30 | 只看该作者
好资料 正在弄红外测距 谢谢分享
回复

使用道具 举报

19#
ID:654753 发表于 2019-12-2 22:54 | 只看该作者
学习中,谢谢分享!
回复

使用道具 举报

20#
ID:475993 发表于 2019-12-4 12:04 | 只看该作者
写的很谢谢,谢谢分享
回复

使用道具 举报

21#
ID:652804 发表于 2020-1-3 09:18 | 只看该作者
好资料,正需要。谢谢楼主。
回复

使用道具 举报

22#
ID:685737 发表于 2020-1-13 17:01 | 只看该作者
32呢?
回复

使用道具 举报

23#
ID:561958 发表于 2020-3-9 20:47 | 只看该作者
学习下,正需要
回复

使用道具 举报

24#
ID:706004 发表于 2020-3-10 19:49 | 只看该作者
学到了,很需要
回复

使用道具 举报

25#
ID:660123 发表于 2020-3-23 10:41 | 只看该作者
请问用GP2Y021替代2D12可以不
回复

使用道具 举报

26#
ID:716087 发表于 2020-3-26 11:39 | 只看该作者
下载资料参考学习
回复

使用道具 举报

27#
ID:660123 发表于 2020-4-30 20:20 | 只看该作者
那个提到的论述Excel表格的文档没找到
回复

使用道具 举报

28#
ID:775865 发表于 2020-6-11 09:48 | 只看该作者
问一下楼主,这个gp2d12在哪能买
回复

使用道具 举报

29#
ID:290578 发表于 2020-6-25 12:33 | 只看该作者
下来测试下,谢谢
回复

使用道具 举报

30#
ID:883211 发表于 2022-7-19 14:49 | 只看该作者
mark 学习一下!
回复

使用道具 举报

31#
ID:1041921 发表于 2022-8-12 09:58 | 只看该作者
好资料,正需要。谢谢楼主。
回复

使用道具 举报

32#
ID:169559 发表于 2022-8-14 10:51 | 只看该作者
这是个好东西,谢谢分享!
回复

使用道具 举报

33#
ID:1056217 发表于 2022-12-7 11:07 | 只看该作者
6学到了,谢谢
回复

使用道具 举报

34#
ID:1059801 发表于 2022-12-30 16:20 | 只看该作者
学到了  楼主写代码的好习惯,设计步骤和代码流程清晰易懂。刚好最近在做 很好的资料
回复

使用道具 举报

35#
ID:539681 发表于 2025-4-21 17:44 | 只看该作者
好人好报好好好
回复

使用道具 举报

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

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

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