| 这篇就上篇的post做个后续处理,发现发送之后发送之后返回数据的接收有问题,因为发送返回的发送成功数据和平台返回的数据并不是一包,按之前数据的接收处理方式会有丢包的情况,我对此进行优化,添加了双缓存处理.之前的数据是接受完包就暂停接收,直至数据用了之后才重新开始,这样会导致在数据接收后到没有处理之前的一段时间有数据发进来,就会导致DMA没打开而丢包,对此进行优化处理,使用双缓存区,DMA缓存区一直接收数据,在接收到转存到待处理缓存区,然后继续接收.下面进行程序优化 1.修改结构体view plaincopy to clipboardprint? 
 
 typedef struct{    uint8_t dma_rx_buff[RX_BUF_MAX_LEN];//DMA缓存区    uint8_t rec_rx_buff[RX_BUF_MAX_LEN];//REC缓存区    uint16_t rec_len;    //DMA接收长度    uint16_t dma_len;    //REC接收长度    volatile uint8_t flag;  //用于判断是否接收完 1为接收完}USRAT_RX;
 
 2.修改中断接收函数view plaincopy to clipboardprint? 
 
 void USART1_IRQHandler(void){  /* USER CODE BEGIN USART1_IRQn 0 */  /* USER CODE END USART1_IRQn 0 */  HAL_UART_IRQHandler(&huart1);  /* USER CODE BEGIN USART1_IRQn 1 */  if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET)//产生空闲中断(说明接收完一包)    {        __HAL_UART_CLEAR_IDLEFLAG(&huart1);//清除空闲中断标志(否则会一直不断进入中断)        HAL_UART_DMAStop(&huart1);//停止本次DMA传输        usart1_rx.dma_len  = RX_BUF_MAX_LEN - __HAL_DMA_GET_COUNTER(&hdma_usart1_rx);//计算接收到的数据长度        if(usart1_rx.rec_len>0)//证明数据未处理即还有未完成数据        {            memcpy(&usart1_rx.rec_rx_buff[usart1_rx.rec_len],usart1_rx.dma_rx_buff,usart1_rx.dma_len); //转存到待处理区域            usart1_rx.rec_len += usart1_rx.dma_len;        }        else        {            memcpy(usart1_rx.rec_rx_buff,usart1_rx.dma_rx_buff,usart1_rx.dma_len);//转存到待处理区域            usart1_rx.rec_len = usart1_rx.dma_len;        }        usart1_rx.flag = 1;        if(usart1_rx.rec_len == 0)//防止误入        {            usart1_rx.flag = 0;        }        HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart1_rx.dma_rx_buff, RX_BUF_MAX_LEN);//设置DMA传输,串口2的数据搬运到缓存中        HAL_UART_DMAResume(&huart1);//恢复DMA传输    }  /* USER CODE END USART1_IRQn 1 */}
 
 3.修改清空函数view plaincopy to clipboardprint? 
 
 void Clear_Usart(USRAT_RX *usart_rx)//初始化(清空)串口结构体{    memset(&usart_rx->rec_rx_buff[0], 0, RX_BUF_MAX_LEN);//全写0    usart_rx->rec_len = 0;    usart_rx->flag = 0;}
 
 4.将之前的数据处理中的rx_buff全改为rec_rx_buff,len都改为rec_len 5.稍微修改代码view plaincopy to clipboardprint? 
 
 Start_Count();OLED_Init();AHT10Init();Clear_Usart(&usart1_rx);//初始化串口结构体__HAL_UART_ENABLE_IT(&huart1,UART_IT_IDLE);//使能空闲中断HAL_UART_Receive_DMA(&huart1, (uint8_t*)usart1_rx.dma_rx_buff, RX_BUF_MAX_LEN);//设置DMA传输,串口2的数据搬运到缓存中esp_init();/*电子产品世界*/OLED_ShowCHinese(0u,0u,0);OLED_ShowCHinese(16u,0u,1);OLED_ShowCHinese(32u,0u,2);OLED_ShowCHinese(48u,0u,3);OLED_ShowCHinese(64u,0u,4);OLED_ShowCHinese(80u,0u,5);/**************//* USER CODE END 2 *//* Infinite loop *//* USER CODE BEGIN WHILE */while (1){    static uint16_t num;    num++;    BH1750_Send_Cmd(ONCE_H_MODE);//单次模式    sprintf(buf,"tem:%.1f   ",pv.tem);    OLED_P8x16Str(0u,2u,(uint8_t *)buf);    sprintf(buf,"hum:%d%%   ",pv.hum);    OLED_P8x16Str(0u,4u,(uint8_t *)buf);    sprintf(buf,"lux:%d     ",pv.lux_16);    OLED_P8x16Str(0u,6u,(uint8_t *)buf);    AHT10ReadData(&pv.tem,&pv.hum);    BH1750_Read_Dat(pv.lux);//读取数据    pv.lux_16 = BH1750_Dat_To_Lux(pv.lux);//转换数据    if(num>100)    {        num = 0;        printf("lux:%d\r\n",pv.lux_16);        printf("tem:%.1f\r\nhum:%d%%\r\n",pv.tem,pv.hum);        OneNet_SendData();//上传数据        HAL_Delay(200);//等待平台回复    }    if(usart1_rx.flag)//接收到数据    {        ipd_rx = ESP8266_GetIPD();//解析数据        if(ipd_rx.len>0)//判断是否为网络下发数据        {             printf("\r\n\r\nipd_rx.IPD_buff:%s\r\n\r\n",ipd_rx.IPD_buff);        }        Clear_Usart(&usart1_rx);//清空缓存    }    /* USER CODE END WHILE */    /* USER CODE BEGIN 3 */}
 
 6.烧录看结果 
  返回数据已经完整接收到了 发现,当发送POST数据之后,它能返回一串数据,仔细看,会发现里边有一串时间 
  这就是GMT时间 GMT时间就是英国格林威治时间,也就是世界标准时间,是本初子午线上的地方时,是0时区的区时,与我国的标准时间北京时间(东八区)相差8小时,即晚8小时 比如GMT时间为14:54:11,北京时间就是(15+8):54:11即为23:54:11,下面就通过字符串操作来取出这段时间,并显示在OLED上,因为我POST的速度并不是很快,所以我只截取了小时和分钟 下面上程序view plaincopy to clipboardprint? 
 
 if(usart1_rx.flag)//接收到数据{    ipd_rx = ESP8266_GetIPD();//解析数据    if(ipd_rx.len>0)//判断是否为网络下发数据    {        printf("\r\n\r\nipd_rx.IPD_buff:%s\r\n\r\n",ipd_rx.IPD_buff);        dateadr= strstr((char *)ipd_rx.IPD_buff, "GMT");//找到"GMT"        memcpy(buf,dateadr-9,5);        hour = (buf[0]-0x30)*10+(buf[1]-0x30);        hour+=8;        if(hour>23)            hour-=24;        min  = (buf[3]-0x30)*10+(buf[4]-0x30);        printf("%02d:%02d\r\n",hour,min);        sprintf(buf,"%02d:%02d   EEPW",hour,min);        OLED_P8x16Str(0u,0u,(uint8_t *)buf);    }    Clear_Usart(&usart1_rx);//清空缓存}
 
 思路很简单就是找到GMT的位置,再往前9个字符,然后截取前5个,把小时数加8,超过23就减24,然后显示在OLED上 效果如下 
  http有POST了,当然不能少了GET,下篇搞一个api get获取天气,制作一个简单的桌面天气 
 |