当前位置:   article > 正文

单片机通过WIFI模块(ESP8266)获取网络时间与天气预报_esp8266获取网络时间

esp8266获取网络时间

        前几天发布了开源4.3寸触摸屏的文章 《开源4.3寸触摸屏》,里面有WIFI获取时间和天气预报相关的功能,今天就来介绍一下这个功能是怎样实现的。

        1.底层驱动

        首先,硬件上,单片机通过串口AT指令访问WIFI模块(ESP12S)。具体AT指令怎样是写,可以参考之前的文章《AT指令程序的编写》。上电后,先配置WIFI模块的工作模式等,用到的AT指令如下:

        1. 关闭回显

                ATE0                   

        2. 设置为STA模式

                AT+CWMODE_DEF=1 

        3. 设置WIFI名称和密码

                AT+CWJAP_DEF=”SSID”,”Password” 

        4. 开启DHCP

                AT+CWDHCP_DEF=1,1 

        5. 设置为单连接模式

                AT+CIPMUX=0      

        6. 透传模式

                AT+CIPMODE=1         

        2.获取网络时间

        获取网络时间的方式有两种,对于较新版本的固件,可以直接通过AT指令来获取。(可以通过AT+GMR指令来查询固件的版本。)

        AT+CIPSNTPTIME?

        对于较老版本的固件,不支持该AT指令,只能通过自己链接网络服务器来获取当前时间。具体需要的AT指令如下:

        1.建立TCP链接

                 AT+CIPSTART="TCP","www.beijing-time.org",80

        2.打开传透模式

                AT+CIPSEND

        3.收到‘>’符号后,随便发送个数据

        4.等待接收数据

        接收的数据可以先用网络调试助手来测试,如下,可以看大,数据中包含了当前的日期和时间信息,我们可以先找到字符串中“Data”的位置,再解析后面的数据。数据格式固定,直接按照字符位置解析就行。

        需要注意的是,获取的这个时间是格林威治时间,与北京时间相差8个小时。

当然,也可以通过连接其它NTP时间服务器来获取时间,步骤是类似的。

        3.获取天气预报

        天气的获取就要比时间稍复杂一些,网上有很多可以提供天气预报的服务器,有些可以免费试用,一般都需要注册一个账号,这里以“心知天气”为例进行介绍。步骤如下:

        1.在该网站注册一个账号:

        心知天气 - 高精度气象数据 - 天气数据API接口 - 行业气象解决方案

        注册之后就可以通过其开放的API来获取天气预报。有免费版和付费版,区别在于访问频次和获取的天气信息内容不同。如下:

        2.在控制台查看账号的私钥,这个在程序用要用。

        3.有了这个私钥,我们就可以通过API来获取天气预报了,相关的API可以在这里查询:

未来15天逐日天气预报和昨日天气 · 心知天气

        先用网络调试助手测试一下:

        可以看到,通过TCP连接服务器后,发送GET请求就可以收到天气预报数据,接下来就是解析数据了。

        4.接收的数据是JSON格式的字符串,在单片机中我们可以移植cJSON来进行解析。cJSON是一个开源的C语音解析库,只有cJSON.c、cJSON.h两个文件,直接添加到工程中就可以使用相关函数。程序如下:

  1. void Weather_JSON( )
  2. {
  3. char *data;
  4. cJSON *root;
  5. cJSON *results;
  6. cJSON *last_update;
  7. cJSON *loc_json, *daily_json;
  8. cJSON *forecast_json;
  9. char *loc_tmp, *weather_tmp;
  10. int i = 0;
  11. data = (char*)(Uart1.RxBuf);//接受到的数据
  12. root = cJSON_Parse(data);
  13. if(root)
  14. {
  15. //printf("JSON格式正确:\n%s\n\n",cJSON_Print(root)); //输出json字符串
  16. results = cJSON_GetObjectItem(root, "results");
  17. results = cJSON_GetArrayItem(results,0);
  18. if(results)
  19. {
  20. loc_json = cJSON_GetObjectItem(results, "location"); //得到location键对应的值,是一个对象
  21. loc_tmp = cJSON_GetObjectItem(loc_json, "id") -> valuestring;
  22. //printf("城市ID:%s\n",loc_tmp);
  23. loc_tmp = cJSON_GetObjectItem(loc_json, "name") -> valuestring;
  24. memset(loc_str,0,20);
  25. memcpy(loc_str,loc_tmp,strlen(loc_tmp));
  26. loc_tmp = cJSON_GetObjectItem(loc_json, "timezone") -> valuestring;
  27. //printf("城市时区:%s\n\n",loc_tmp);
  28. daily_json = cJSON_GetObjectItem(results, "daily");
  29. if(daily_json)
  30. {
  31. Weather_Dat[0].Flag = 1;
  32. for(i = 0; i < 3; i++)
  33. {
  34. forecast_json = cJSON_GetArrayItem(daily_json, i);
  35. //weather_tmp = cJSON_GetObjectItem(forecast_json, "date") -> valuestring;//日期
  36. weather_tmp = cJSON_GetObjectItem(forecast_json, "code_day") -> valuestring;//白天天气代码
  37. Weather_Dat[i].Weathcode = atoi(weather_tmp);
  38. //weather_tmp = cJSON_GetObjectItem(forecast_json, "code_night") -> valuestring;//晚上天气代码
  39. weather_tmp = cJSON_GetObjectItem(forecast_json, "high") -> valuestring;//最高温度
  40. memset(Weather_Dat[i].HighT,0,4);
  41. memcpy(Weather_Dat[i].HighT,weather_tmp,strlen(weather_tmp));
  42. weather_tmp = cJSON_GetObjectItem(forecast_json, "low") -> valuestring;//最低温度
  43. memset(Weather_Dat[i].LowT,0,4);
  44. memcpy(Weather_Dat[i].LowT,weather_tmp,strlen(weather_tmp));
  45. weather_tmp = cJSON_GetObjectItem(forecast_json, "wind_direction") -> valuestring;//风向
  46. memset(Weather_Dat[i].Wind_Dir,0,10);
  47. memcpy(Weather_Dat[i].Wind_Dir,weather_tmp,strlen(weather_tmp));
  48. weather_tmp = cJSON_GetObjectItem(forecast_json, "wind_scale") -> valuestring;//风力
  49. memset(Weather_Dat[i].WindScale,0,4);
  50. memcpy(Weather_Dat[i].WindScale,weather_tmp,strlen(weather_tmp));
  51. weather_tmp = cJSON_GetObjectItem(forecast_json, "humidity") -> valuestring;//湿度
  52. memset(Weather_Dat[i].Humi,0,4);
  53. memcpy(Weather_Dat[i].Humi,weather_tmp,strlen(weather_tmp));
  54. }
  55. }
  56. else
  57. {
  58. //printf("daily json格式错误\r\n");
  59. Weather_Dat[0].Flag = 0;
  60. }
  61. // last_update = cJSON_GetObjectItem(results, "last_update");
  62. // update_tmp = last_update->valuestring;
  63. // if(last_update)
  64. // {
  65. // //printf("更新时间:%s\r\n", update_tmp);
  66. // }
  67. }
  68. else
  69. {
  70. Weather_Dat[0].Flag = 0;
  71. //printf("results格式错误:%s\r\n", cJSON_GetErrorPtr());
  72. }
  73. }
  74. else
  75. {
  76. Weather_Dat[0].Flag = 0;
  77. //printf("JSON格式错误\r\n");
  78. }
  79. cJSON_Delete(root);
  80. }

        解析后,将有用数据转换为自己想要的格式存储在变量中。需要注意的是,cJSON可能占用较大的堆栈,因此堆栈需要设大一些。

        总结一下,获取天气预报的步骤如下:

        1.注册账号

        2.获取私钥

        3.连接服务器

                AT+CIPSTART="TCP”,” api.seniverse.com”,80

        4..打开传透模式

            AT+CIPSEND

        5..收到‘>’符号后,发送GET请求

GEThttps://api.seniverse.com/v3/weather/daily.json?key=your_key&location=ip&language=zh-Hans&unit=c&start=0&days=3

     6.等待接收数据,并解析

        以上就是单片机通过AT指令访问WIFI模块获取网络时间和天气预报的方法,需要注意的是上述所有指令后面别忘了加回车换行符。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/482276
推荐阅读
相关标签
  

闽ICP备14008679号