Esp8266 进阶之路20 【高级篇】深入学习esp8266的esp now模式组网,仿机智云做一个小网关,实现无需网络下轻松彼此连接通讯交互数据。(附带Demo)


目录:


一.前言;

  • 有十天没发博文了,心里难受!高考来了,祝各位考生取得理想成绩!最近又在逛机智云的论坛,看看有没有新的技术可以学习。发现有个小网关的设备,也是基于esp8266的,认真地看了看机智云对此网关的定义:

1、子设备需要在云端注册(目前选择单品类产品注册),子设备数据点当然可以用,中控其中一个重要的功能就是透传数据点数据(上行、下行)。
2、子设备和云端没有直接的物理通道,所以子设备和云端和APP之间的通信都是由网关代理完成。
3、子设备与网关的通信,机智云不参与设计,这部分属于私有通信协议,使用Gateway SDK的开发者我们给通信层留出接口API,按照API实现网关的通信层开发即可(比如,zigbee我们不关心主从节点之间的通信协议)。

  • 以我个人理解,总的来说就是:小网管支持控制多个子设备,而子设备可以单独和服务器通讯。子设备和服务器的通讯协议是机智云提供的,而子设备和网关的协议是自己定的。而我现在要做的是也是如此,只是我们可以独立出来子设备和网关设备,可以做到协议都自定 。

这里写图片描述


二. 走进乐鑫的esp-now技术领域;


  • ①:下面是乐鑫对其的定义:

这里写图片描述


  • ②:可看到,此技术和组网Mesh非常相似,也是类似自组网通讯。非常适合那些小数据的通讯,比如智能灯、温湿度传感器的采集传输,类似ZigBee

  • ③:至于底层原理是怎么实现?我们也不需要怎么理会。只需要自己去调用乐鑫提供的接口即可,下面是几个重要的接口方法说明:

//初始化
int esp_now_init(void);
//删除
int esp_now_deinit(void);

//注册ESP-NOW发包函数回调
int esp_now_register_send_cb(esp_now_send_cb_t cb);
//注销ESP-NOW发包函数回调
int esp_now_unregister_send_cb(void);

//注册ESP-NOW接收函数回调
int esp_now_register_recv_cb(esp_now_recv_cb_t cb);
//注销ESP-NOW接收函数回调
int esp_now_unregister_recv_cb(void);

//发送一数据到指定的设备
int esp_now_send(u8 *da, u8 *data, int len);

//添加一个对等体设备
int esp_now_add_peer(u8 *mac_addr, u8 role, u8 channel, u8 *key, u8 key_len);
//删除一个对等体设备
int esp_now_del_peer(u8 *mac_addr);

//设置自己的角色
int esp_now_set_self_role(u8 role);
//获取自己的角色
int esp_now_get_self_role(void);

  • ④:上面理解可知,分为主角色和从角色,这就相当于我们的小网关和子设备!下面是一些信息的解析说明:

    • 注意点一:发送的数据包最大支持 250字节的 payLoad ;
    • 注意点二:加密配对设备有限制,station模式支持10个加密配对设备;在softAP模式或者softAPstation模式共存下支持6个加密设备配对。而非加密设备的配对理论支持无限个,但在和加密设备总数不超过20个;
    • 注意点三:channle信息理解为消息所在的信道,0到255范围,这类似2.4g模块的信道。通讯所需要!
    • 注意点四:上面提到的加密设备在esp_now_add_peer()方法中调用,此方法顾名思义就是增加对方设备在自己的应用层。

这里写图片描述


这里写图片描述


三. 代码实现;


  • 贴代码之前,必须理清下思路:


    • ①:不管是子设备发送消息到主设备,或者是主设备发送设备到子设备,都必须要有统一的channle,如果加密配对,必须要有一致的Key, 因此网关的实现不是自动组网的,而是需要手动地添加对应的设备mac地址;只有拿到了这个地址,后面的通讯风雨无阻。
    • ②:说到必须要有mac地址(注意是station的地址,不是softAP地址,这可以通过代码获取或者看烧录工具提示!) ,那么我们要通过某个途径来把对面的地址存在主设备中,这个方法很多种:比如先让子设备配网先,让APP去把此设备的地址发给中控设备,让中控设备记录此mac地址,保存在flash中!
    • ③:已经拿到了多个子设备的地址后,中控设备就可以按照上位机下发的提示去做统一的协议下发给子设备,如果要调用发送给所有的子设备,在发送函数中传入NULL即可,注意若匹配设备过多,会造成阻塞,丢包现象。
    • ④:由于此研究较为严谨,涉及到商业机密,我就不公开全部代码了,下面提供的源码都是可以做下毕设或者个人diy的,如果有兴趣,加群一起讨论吧!

小网关和子设备的核心代码:

  • 代码一样,只是需要区分哪个是主动发送消息,以及实际上我们的子设备工程还要处理其他逻辑,比如智能灯的pwm输出;但是小网关设备仅仅做和服务器交互数据为主,如果家庭路由器不想接那么多子设备,或者信号弱,直接把一个小网关放在家庭中心,做数据转发,大大减轻了家庭的路由器负担,何乐而不为?

  • ①:连接家庭路由器,此步骤可有可无,但是必须要station模式:
    wifi_set_opmode(STATION_MODE);  //设置为STATION模式
    struct station_config stationConf;
    os_strcpy(stationConf.ssid, "iPhone");    //改成你自己的   路由器的用户名
    os_strcpy(stationConf.password, "xh870189248"); //改成你自己的   路由器的密码
    wifi_station_set_config(&stationConf);  //设置WiFi station接口配置,并保存到 flash
    wifi_station_connect(); //连接路由器
  • ②:初始化esp_now,注册 ESP-NOW 收包的回调函数和发包回调函数:

    if (esp_now_init() == 0) {
        os_printf("esp_now init ok\n");

        // 注册 ESP-NOW 收包的回调函数
        esp_now_register_recv_cb(user_esp_now_recv_cb);
        // 注册发包回调函数
        esp_now_register_send_cb(user_esp_now_send_cb);
    }
  • ③:设置角色位置,这里我设置为双重角色,因为下面的子设备为slave角色,从station发包不成功:
        esp_now_set_self_role(ESP_NOW_ROLE_COMBO);
        esp_now_add_peer(slave_mac, ESP_NOW_ROLE_COMBO, 1, NULL, 16);
  • ④:发包给指定的设备,注意传的是station模式的mac地址,不是ip地址!:
user_esp_now_send(mac, send_data, os_strlen(send_data));

四. 串口打印;


  • 截图模糊,放大浏览:

这里写图片描述


五. 其他;


  • 怎么通过下载工具查看地址,截图模糊,放大浏览:

这里写图片描述

这里写图片描述

猜你喜欢

转载自blog.csdn.net/xh870189248/article/details/80631739