基于ZIGBEE和ESP8266的智能生态系统(三)

ESP8266篇

esp8266作为一个智能家居,物联网最基本的平台,可以说是家喻户晓了。无论是产品还是自己做一些简单的小项目,用这个都还是比较合适的。8266主要可以用AT指令控制,也可以用自带的SDK开发,官方已经把开发环境做好了,上手起来还是比较方便的。能用SDK当然就不建议玩AT了,关于SDK方面的学习,我自己就不班门弄斧了,各位可以参考下面的链接,博主也是这样学过来的。
https://www.cnblogs.com/imliubo/p/10090798.html

然后博主是基于机智云来做的,作为不会写app的小伙伴,这种官方提供好的平台还是很有用处,根据自己需求配置,一些小项目还是足够了。
机智云教程请参考如下链接。
https://blog.csdn.net/weixin_43353164/article/details/82961659

步入正题,之前把zigbee部分都做完了,现在来做esp8266的,这方面很简单,搭好基础矿建后,根据需求改代码就行了。
首先我定义了两个结构体,分别用来存放我两个zigbee发送过来的数据。

typedef struct {
	bool led;
	bool fan;
	uint32_t temp;
	uint32_t humid;
	uint32_t light;
	uint32_t fan_start;
	uint32_t led_start;
}ZigBee;
ZigBee Place_A =
{
	.led = 0,
	.fan = 0,
	.temp = 0,
	.humid = 0,
	.light = 0,
	.fan_start = 0,
	.led_start = 0,
};
ZigBee Place_B =
{
	.led = 0,
	.fan = 0,
	.temp = 0,
	.humid = 0,
	.light = 0,
	.fan_start = 0,
	.led_start = 0,
};

进入userHandle函数,在此我们将zigbee发过来的数据存入机智云当前显示的变量中。

void ICACHE_FLASH_ATTR userHandle(void)
{
	static uint8 ledtime = 0;

	ledtime++;

	if (LED_TIMEOUT < ledtime)
	{
	    ledtime = 0;
		currentDataPoint.valueA_LED 	  = Place_A.led;
		currentDataPoint.valueA_FAN 	  = Place_A.fan;
		currentDataPoint.valueA_temp 	  = Place_A.temp;
		currentDataPoint.valueA_humid 	  = Place_A.humid;
		currentDataPoint.valueA_light 	  = Place_A.light;
		currentDataPoint.valueA_fan_start = Place_A.fan_start;
		currentDataPoint.valueA_led_start = Place_A.led_start;

		currentDataPoint.valueB_LED 	  = Place_B.led;
		currentDataPoint.valueB_FAN 	  = Place_B.fan;
		currentDataPoint.valueB_temp 	  = Place_B.temp;
		currentDataPoint.valueB_humid 	  = Place_B.humid;
		currentDataPoint.valueB_light 	  = Place_B.light;
		currentDataPoint.valueB_fan_start = Place_B.fan_start;
		currentDataPoint.valueB_led_start = Place_B.led_start;
	}

    system_os_post(USER_TASK_PRIO_2, SIG_UPGRADE_DATA, 0);
}

进入gizwitsEventProcess函数,通过8266控制zigbee的部分都在此完成,这里只贴了基本的代码,剩下的变量以此类推完成。

		case EVENT_LED :
            currentDataPoint.valueLED = dataPointPtr->valueLED;
            GIZWITS_LOG("Evt: EVENT_LED %d \n", currentDataPoint.valueLED);
            if(0x01 == currentDataPoint.valueLED)
            {
            	uart0_sendStr("manual");
            }
            else
            {
            	uart0_sendStr("auto");
            }
            break;
        case EVENT_A_LED :
            currentDataPoint.valueA_LED = dataPointPtr->valueA_LED;
            GIZWITS_LOG("Evt: EVENT_A_LED %d \n", currentDataPoint.valueA_LED);
            if(0x01 == currentDataPoint.valueA_LED)
            {
            	uart0_sendStr(A_LED_ON);
            	Place_A.led = 1;
            }
            else
            {
            	uart0_sendStr(A_LED_OFF);
            	Place_A.led = 0;
            }
            break;
		case EVENT_A_fan_start:
            currentDataPoint.valueA_fan_start= dataPointPtr->valueA_fan_start;
            GIZWITS_LOG("Evt:EVENT_A_fan_start %d\n",currentDataPoint.valueA_fan_start);
            Place_A.fan_start = currentDataPoint.valueA_fan_start;
            os_sprintf(buf, "A_FAN_START:%d", currentDataPoint.valueA_fan_start);
            uart0_sendStr(buf);
            break;

接下来我们做接收数据并处理的部分。
首先8266的串口先要做一些处理,让串口一用作系统默认的打印信息,而串口0用于和zigbee通信。

uart_init(UartBautRate uart0_br, UartBautRate uart1_br)
{
    /*this is a example to process uart data from task,please change the priority to fit your application task if exists*/
    system_os_task(uart_recvTask, uart_recvTaskPrio, uart_recvTaskQueue, uart_recvTaskQueueLen);  //demo with a task to process the uart data

    UartDev.baut_rate = uart0_br;
    uart_config(UART0);
    UartDev.baut_rate = uart1_br;
    uart_config(UART1);
    ETS_UART_INTR_ENABLE();

    #if UART_BUFF_EN
    pTxBuffer = Uart_Buf_Init(UART_TX_BUFFER_SIZE);
    pRxBuffer = Uart_Buf_Init(UART_RX_BUFFER_SIZE);
    #endif
    
    #if UART_SELFTEST&UART_BUFF_EN
    os_timer_disarm(&buff_timer_t);
    os_timer_setfn(&buff_timer_t, uart_test_rx , NULL);   //a demo to process the data in uart rx buffer
    os_timer_arm(&buff_timer_t,10,1);
    #endif
}

system_os_task(uart_recvTask, uart_recvTaskPrio, uart_recvTaskQueue, uart_recvTaskQueueLen);
这个函数是创建一个任务,就是用如处理串口0的接收数据的,
uart_config(UART0);
这就是配置串口寄存器,在这个里面有设置了串口的回调函数

关于串口不明白的直接看下面链接,讲8266串口接收的。
https://blog.csdn.net/chen244798611/article/details/51470630/

然后在user_init里加上

 	uart_init(115200,115200);
    UART_SetPrintPort(1);

然后我们就进入回调函数中处理数据。

LOCAL 	 ICACHE_FLASH_ATTR ///
uart_recvTask(os_event_t *events)
{
	uint8 BUF[100];
	uint8 buf[10];
	uint8 j = 0;
	uint8 i = 3;
    if(events->sig == 0){
    #if  UART_BUFF_EN
        Uart_rx_buff_enq();
    #else
        uint8 fifo_len = (READ_PERI_REG(UART_STATUS(UART0))>>UART_RXFIFO_CNT_S)&UART_RXFIFO_CNT;
        uint8 d_tmp = 0;
        uint8 idx=0;
        for(idx=0;idx<fifo_len;idx++) {
            d_tmp = READ_PERI_REG(UART_FIFO(UART0)) & 0xFF;
            BUF[j++] = d_tmp;
            //uart_tx_one_char(UART0, d_tmp);
        }
        //uart0_sendStr(BUF);
        if (BUF[0] == '@')
        {
        	if (BUF[1] == 'A')
			{
				Place_A.led 	  = BUF[2];
				Place_A.fan 	  = BUF[3];
				Place_A.temp 	  = data_conver(&i, BUF);
				Place_A.humid 	  = data_conver(&i, BUF);
				Place_A.light 	  = data_conver(&i, BUF);
				Place_A.fan_start = data_conver(&i, BUF);
				Place_A.led_start = data_conver(&i, BUF);
			}

			else if (BUF[1] == 'B')
			{
				Place_B.led 	  = BUF[2];
				Place_B.fan 	  = BUF[3];
				Place_B.temp 	  = data_conver(&i, BUF);
				Place_B.humid 	  = data_conver(&i, BUF);
				Place_B.light 	  = data_conver(&i, BUF);
				Place_B.fan_start = data_conver(&i, BUF);
				Place_B.led_start = data_conver(&i, BUF);
			}
        }


        WRITE_PERI_REG(UART_INT_CLR(UART0), UART_RXFIFO_FULL_INT_CLR|UART_RXFIFO_TOUT_INT_CLR);
        uart_rx_intr_enable(UART0);
    #endif
    }else if(events->sig == 1){
    #if UART_BUFF_EN
	 //already move uart buffer output to uart empty interrupt
        //tx_start_uart_buffer(UART0);
    #else

    #endif
    }
}
```c
data_conver(&i, BUF)是一个数据处理函数,根据用户自定义的数据格式来解析数据。

ICACHE_FLASH_ATTR
int abc(int data, int n)
{
	int x = data;

	if (n == 0)
	{
		return 1;
	}

	while(--n)
	{
		data *= x;
	}

	return data;
}

ICACHE_FLASH_ATTR
uint32 data_conver(uint8 *i, uint8 buf[])
{
	uint32 data = 0;
	uint8 temp = *i;
	uint8 j;

	while(buf[++(*i)] != '#');

	for (j = *i-temp-1; j > 0; j--)
	{
		data += (buf[++temp] - '0') * abc(10, j-1);
	}

	return data;
}

这样ESP8266方面就全部做完了,是不是很简单。
接下来我们来看一下效果。
在这里插入图片描述
在这里插入图片描述
因为之前玩坏了一块,所以就用一块来试验的,效果是还不错的,也可以远程控制,不过我只用LED测试了一下,继电器什么的都还没有装。

总结

zigbee和esp8266做智能家居和物联网这些实在是不错,自己买来玩一玩也可以做个小东西。博主也面试过几个做物联网和智能家居的,不过公司的话涉及的会更底层,而不是这么简单的应用,如果有意向往这方面发展的小伙伴还是可以考虑深入学习一下。

本文首发于DF创客社区作者:2877137721 原文链接: DIY属于你的智能家居系统,zigbee,esp8266,51单片机 DIY家庭智能家居控制系统(2)51,zigbeeesp8266 相关附件于原文下方下载 【介绍】 随着微功耗处理器以及通讯芯片的发展,以往较为耗能的有线通讯方式越来越阻碍了通讯网络的发展,于是一大批的无线通讯方案应运而生,例如NBIOT,蓝牙4.0,zigbee等等。同时微功耗无线处理器的流行使得传统家电带上了智能的色彩,智能家居必将是未来家庭的必备。某些高科技企业也在揶揄这块市场,纷纷推出了自家的智能家居解决方案,比较知名的有米家方案以及阿里智能的解决方案。但是回到现实的使用上来说,每个家庭的条件环境其实不一样,同样的产品未必在每个家庭上都可以使用方便,所以个性化的定制产品才可以更好的方便我们的生活。 于是在暑假期间,萌生了DIY一套智能家居方案的想法,具有各种智能开关和传感器等节点,并且对接物联平台实现联网控制以及数据的上传。由于家庭中将会使用到的智能设备种类繁多,所以我将采用模块化的设计思路,即采用核心板加外围功能部分的思路,像搭积木一样的构建各个智能设备。大家有同样想制作的想法可以参考。 截至发帖前,我完成了智能网关,智能墙壁开关,无线遥控开关以及无线气象站的设计制作,之后有新的设备加入的话,我会及时的更新。 【准备事项】 完成这个涉及到多方面的项目,需要的硬件设备以及开发环境较多,所以我先大概给出一个列表: 软件篇: ArduinoIDE:用于给ESP8266编程使用 KEIL uVision5 :用于给51单片机编程使用 串口调试器:用于监控单片机输出数据 lceda设计软件:绘制电路原理图,设计PCB电路板 硬件篇: NodeMCU开发板一块 亿佰特zigbee模块若干(视节点数量而定) STC8F2系列单片机若干 核心的主要控制器件在此列出,其他元器件会在制作过程中一一说明。 【制作过程】 我构想了一张智能家庭的网络拓扑图,大家可以欣赏以下,后期图上的设备都将加入进来: 由于涉及到多个设备的设计制作,所以本个报名贴先奉上智能家居核心板,智能网关以及无线气象站的制作过程: #制作过程之核心板篇# >>>本项目使用zigbee网络进行智能家居之间的通信。 好早之前就接触过zigbee组网,对于这种低成本低功耗的网络还是抱有很大信心的。这里讲解选择这个网络的几大理由: mesh结构的网络很适合智能家居的控制结构,在入网的任何一个节点都可以访问到所有节点的数据,这点很适合网关控制各个设备。 低功耗使得终端设备甚至可以采用电池供电,使得所有的模组都尽量可能的无线化。 多跳传输,无线方案中最大的问题就是数据发送的不稳定以及障碍物对信号的遮挡导致数据无法正确传输,而ZigBee的多跳恰好解决了这个问题,节点会自动选择优质的传输路线多跳传输,保证信号质量。 总结一下:zigbee网络确实是好,但是对于我这种比较懒惰的人并不想去学习那复杂的传输理论以及zigbee通讯芯片的编程,于是我选择上网购买ZigBee模块,最后选择了一种小型的串口转zigbee模块,比较方便单片机通讯。 说到模块化,那么就需要核心的控制板。本质来说就是将单片机以及购买的zigbee模块集成到一张电路板上,并且预留各种接口,方便后期移植到各种设备上,这样一张电路板就可以适用各种智能家居设备了。 在保证功能足够的同时,体积也是我需要考虑的问题,如果核心板做的过大,会导致无法安装到某些空间狭小的智能设备中,所以小体积是我所着重考虑的。 基于多方面考虑之后,stc8F2k08s2进入了我的视线,小巧的sop16封装以及简单的外围电路,足够的IO口(14个)。这款芯片成为了核心板的控制单片机。于是一番绘制原理图以及PCB之后,成品大概明朗了: 焊接好的成品如下: 这将会成为以后所有我的智能家居方案的核心控制板。 #制作过程之智能网关篇# 完成了核心板的制作,网关成为了下一个比较关注的对象,因为家中的所有智能设备的控制以及通讯都会由他来完成,包括链接到互联网上传数据,所以说对于网关的硬件选取也是重中之重。上联互联网,下接zigbee小型通讯网,网关担任了一个家庭控制中心的角色,对于它的选择我认为esp8266是个不错的方案,可以支持arduino ide编程,这对于不太了解网络通讯协议的我来说是个好消息,因为在arduino的编程中,我可以借助强大的库函数来完成我想要的功能。 ESP8266是一个拥有了近80Mhz的主频的32位处理器,内置了wifi模组省去了网线,并且拥有丰富的外设以及较高的性能,可以胜任智能网关的工作。当然,近期乐鑫发布的ESP32系列芯片也会是个好的选择,更高的主频以及wifi蓝牙双模的设计让其拥有更加方便的接
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值