>[success] 導師視頻講解:[去聽課](https://www.bilibili.com/video/BV1k34y1D7Vz/)
>[success] **技術支持說明:**
>**1**.一般以自主學習為主
> **2**.可到官方問答社區中提問:[**去提問**](https://bbs.csdn.net/forums/zigbee)
> **3**.工程師**會盡快**解答社區問題,但他們是一線開發,【**難以保證**】解答時效,解答辛苦,感謝理解!
<br/>
本節課將以實驗的方式講解如何使用ZCL讀寫命令API,實驗設備包含一個協調器和一個終端,內容是協調器向終端發送寫命令,然后在再發送讀命令,最后比較一下寫入的信息和讀取到的信息是否一致。
<br/>
## **定義讀寫事件**
在zcl\_samplesw.h文件中分別定義一個讀命令事件和寫命令事件,代碼如下:

###
協調器接收到終端設備廣播(Annce)的信息后,啟動一個讀命令事件,代碼如下:

###
在zcl\_samplews.c文件的zclSample\_event\_loop事件處理函數中,可以找到讀命令事件的處理代碼,如下:
~~~
//
if ( events & SAMPLEAPP_READ_EVT )//如是讀命令事件
{
zclSampleSw_ReadTest();//讀命令事件處理函數
//啟動一個寫命令事件
osal_start_timerEx(zclSampleSw_TaskID,
SAMPLEAPP_WRITE_EVT,
SAMPLEAPP_WRITE_PERIOD);
return ( events ^ SAMPLEAPP_READ_EVT );
}
~~~
###
讀命令事件處理函數zclSampleSw\_ReadTest的代碼定義如下:
###
```
1.static?void?zclSampleSw_ReadTest(void)??
2.{??
3.????afAddrType_t?destAddr;??
4.????zclReadCmd_t?*readCmd;??
5.????static?uint8?txID?=?0;??
6.??????
7.????destAddr.endPoint?=?SAMPLESW_ENDPOINT;??
8.????destAddr.addrMode?=?afAddr16Bit;??
9.????destAddr.addr.shortAddr?=?zclSampleSw_TestAddr;??
10.?
//申請一個動態內存
11.? readCmd?=?(zclReadCmd_t?*)osal_mem_alloc(sizeof(zclReadCmd_t)?+??? sizeof(uint16));
12.
13.???if(readCmd?==?NULL)//判斷是否成功申請到內存
14.??????return;??
15.????readCmd->numAttr?=?1;//待讀取的屬性數量為1
16.????readCmd->attrID[0]?=?ATTRID_ON_OFF_SWITCH_ACTIONS;//待讀取的屬性ID
17.??????
18.????zcl_SendRead(SAMPLESW_ENDPOINT,??
19.?????????????????&destAddr,??
20.?????????????????ZCL_CLUSTER_ID_GEN_ON_OFF_SWITCH_CONFIG,//Cluster ID??
21.?????????????????readCmd,??
22.?????????????????ZCL_FRAME_CLIENT_SERVER_DIR,//通信方向是由客戶端到服務器端
23.?????????????????TRUE,
24.?????????????????txID++);??
25. osal_mem_free(readCmd);//釋放內存
26.}
```
###
在讀命令處理中代碼中還啟動一個寫命令事件,在讀命令事件處理代碼的下方可以找到寫命令事件的處理代碼,如下:
~~~
if ( events & SAMPLEAPP_WRITE_EVT )//如果是寫命令事件
{
zclSampleSw_WriteTest();//寫命令處理函數
osal_start_timerEx(zclSampleSw_TaskID,//啟動一個讀命令事件
SAMPLEAPP_READ_EVT,
SAMPLEAPP_READ_PERIOD);
return ( events ^ SAMPLEAPP_WRITE_EVT );
}
~~~
###
寫命令事件處理函數zclSampleSw\_WriteTest的代碼定義如下:
###
```
static?void?zclSampleSw_WriteTest(void)??
{??
????afAddrType_t?destAddr;??
????zclWriteCmd_t?*writeCmd;??
????static?uint8?txID?=?0;??
??????
????destAddr.endPoint?=?SAMPLESW_ENDPOINT;??
????destAddr.addrMode?=?afAddr16Bit;??
????destAddr.addr.shortAddr?=?zclSampleSw_TestAddr;??
??????
? writeCmd=(zclWriteCmd_t?*)osal_mem_alloc(sizeof(zclWriteCmd_t)?+??? sizeof(zclWriteRec_t));//申請一個動態內存
??? if(writeCmd?==?NULL)//判斷動態內存是否申請成功
???? return;??
??????
writeCmd->attrList[0].attrData=(uint8*)osal_mem_alloc(sizeof(uint8));//申請一個動態內存????
if(writeCmd->attrList[0].attrData?==?NULL)//判斷動態內存是否申請成功??
???? return;??
??????
????writeCmd->numAttr?=?1;//待寫入的屬性數量
????writeCmd->attrList[0].attrID?=ATTRID_ON_OFF_SWITCH_ACTIONS;待寫入的屬性的ID
????writeCmd->attrList[0].dataType?=?ZCL_DATATYPE_ENUM8;//屬性值的類型
????*(writeCmd->attrList[0].attrData)?=?txID;//屬性值
????HalLcdWriteStringValue("Write:",?txID,?10,?4);??
????
????zcl_SendWrite(SAMPLESW_ENDPOINT,??
&destAddr,??
ZCL_CLUSTER_ID_GEN_ON_OFF_SWITCH_CONFIG,//Cluster ID?
writeCmd,??
ZCL_FRAME_CLIENT_SERVER_DIR,//通信方向是由客戶端到服務器端
TRUE,
txID++);??
osal_mem_free(writeCmd->attrList[0].attrData); // 釋放內存
osal_mem_free(writeCmd); // 釋放內存
}
```
<br/>
## **讀響應處理**
在客戶端中處理從服務器端中讀取到的信息,即編寫讀命令響應信息處理函數,代碼如下:
~~~
#ifdef ZCL_READ
/*********************************************************************
* @fn zclSampleSw_ProcessInReadRspCmd
*
* @brief 讀響應處理函數
*
* @param pInMsg - 待處理的消息
*
* @return
*/
static uint8 zclSampleSw_ProcessInReadRspCmd( zclIncomingMsg_t *pInMsg )
{
zclReadRspCmd_t *readRspCmd;
uint8 i;
readRspCmd = (zclReadRspCmd_t *)pInMsg->attrCmd;
for (i = 0; i < readRspCmd->numAttr; i++)//readRspCmd->numAttr為屬性的數量
{
if( pInMsg->clusterId == ZCL_CLUSTER_ID_GEN_ON_OFF_SWITCH_CONFIG &&//如果該消息是關于指定的Cluster
readRspCmd->attrList[i].attrID == ATTRID_ON_OFF_SWITCH_ACTIONS )//如果該屬性的ID是指定的屬性ID
{
uint8 val;
val = *(readRspCmd->attrList[i].data);//讀取屬性值
HalLcdWriteStringValue("Read:", val, 10, 4);//顯示信息到屏幕中
}
}
return TRUE;
}
#endif // ZCL_READ
~~~
<br/>
## **仿真調試**
* 分別編譯協調器和終端(路由器)工程,然后分別下載到兩個開發板中。
* 終端(路由器)設備加入到ZigBee網絡中后,可以看到協調器屏幕顯示如下提示信息。


可以觀察到讀和寫的數據是一樣的。
<br/>
<br/>
## **項目定制**
* 如需項目定制開發,可掃碼添加項目經理好友(注明“**項目定制**”)
* 定制范圍:**NB-IoT**、**CATn(4G)**、**WiFi**、**ZigBee**、**BLE Mesh**以及**STM32**、**嵌入式Linux**等IoT技術方案
* 善學坊官網:[www.sxf-iot.com](https://www.sxf-iot.com/)

* 非項目定制**勿擾**,此處**非**技術支持
- 課程簡介
- 配套資源下載
- 配套開發套件簡介
- 簡介
- 技術參數
- 電路原理圖 & PCB圖——標準板
- 電路原理圖 & PCB圖——MiNi板
- CC2530F256 核心模組
- MCU簡介
- 模組尺寸 & 引腳定義
- 模組技術參數
- 電路原理圖 & PCB設計圖
- 封裝及生產指導
- 第一部分:準備
- 1.1 小白也能讀懂的 ZigBee 3.0 簡介
- 1.2 IAR EW for 8051 簡介與安裝
- 1.3 TI Z-Stack 3.0 簡介與安裝
- 1.4 SmartRF Flash Programmer 下載與安裝
- 1.5 串口助手簡介與安裝
- 1.6 SmartRF04EB 驅動程序
- 1.7 USB轉串口驅動程序
- 其他軟件安裝(非必須)
- 1.7.1 Xshell 7 簡介與安裝指南
- 1.7.2 PuTTY 簡介與安裝
- 第二部分:51單片機入門——基于CC2530
- 第1章:CC2530 開發基礎實驗
- 1.1 新建工作空間及工程
- 1.2 源代碼編寫及編譯
- 1.3 程序下載及仿真
- 1.4 固件燒錄
- 第2章:GPIO實驗
- 2.1 多工程管理基礎
- 2.2 GPIO輸出實驗——LED控制
- 2.3 GPIO輸入實驗——機械按鍵
- 2.4 GPIO輸入輸出通用配置實驗
- 2.5 GPIO外部中斷實驗
- 第3章:定時器實驗
- 3.1 工程概述
- 3.2 定時器T1實驗——查詢觸發
- 3.3 定時器T3實驗——中斷觸發
- 3.4 看門狗定時器實驗
- 3.5 低功耗定時器實驗
- 第4章:串口通信實驗
- 第5章:ADC實驗——使用光照傳感器
- 第6章:OLED 顯示器實驗
- 第7章:外設實驗
- 7.1 DHT11溫濕度傳感器
- 7.2 NorFLASH讀寫實驗
- 7.3 繼電器控制實驗
- 第三部分:Z-Stack 3.0 詳解
- 第1章:Z-Stack 3.0 架構詳解
- 1.1 Z-Stack 3.0.1 文件組織
- 2.2 Z-Stack 3.0.1 工程框架
- 第2章:操作系統的任務調度原理
- 第3章:OSAL 詳解
- 3.1 OSAL的任務調度原理
- 3.2 任務初池始化與事件處理
- 3.3 Z-Stack 事件的應用
- 3.4 使用動態內存
- 第4章:硬件適配層應用——LED
- 4.1 HAL的文件結構和工程結構
- 4.2 HAL的架構簡介
- 4.2 LED API簡介
- 4.3 LED 實驗
- 第5章:硬件適配層應用——按鍵
- 5.1 按鍵實驗
- 5.2 HAL 按鍵框架詳解(選修)
- 第6章:硬件適配層應用——串口
- 第7章:硬件適配層應用——顯示屏
- 第8章:硬件適配層應用——ADC
- 第四部分:ZigBee 3.0 網絡編程
- 第1章:ZigBee 3.0 網絡原理
- 1.1 協議層次結構
- 1.2 IEEE 802.15.4協議
- 1.3 網絡層
- 第2章:ZigBee 3.0 BDB
- 2.1 BDB 簡介
- 2.2 BDB Commissioning Modes
- 2.3 ZigBee 3.0 組網實驗
- 第3章:基于AF的數據通信
- 3.1 簡單描述符
- 3.2 通信原理
- 3.3 數據發送API簡介
- 3.4 ZigBee 3.0 通信實驗
- 第4章:ZCL 基本原理
- 4.1 ZCL 簡介
- 4.2 ZCL 內容詳解
- 第5章:基于ZCL的開關命令收發
- 5.1 應用層對 ZCL API 的調用
- 5.2 ZCL 開關命令收發 API
- 5.3 ZCL 開關命令收發實驗
- 第6章:基于ZCL的屬性讀寫
- 6.1 ZCL 屬性讀寫 API
- 6.2 ZCL 屬性讀寫實驗
- 第7章:基于ZCL的屬性上報實驗
- 7.1 概述
- 7.2 終端設備開發
- 7.3 協調器設備開發
- 7.4 仿真調試
- 課外篇:項目實戰
- ZigBee 3.0 環境信息采集
- 基于ZigBee的農業環境信息采集
- 基于ZigBee的文件傳輸系統
- 基于ZigBee的光照自動開關窗簾
- 基于ZigBee的溫濕度 & 光照強度采集系統
- 其他項目
- 基于ZigBee的溫度和有害氣體短信報警系統
- 基于ZigBee的多傳感器探測與亮燈報警系統
- 基于ZigBee的溫濕度、人體紅外與聲光報警系統
- ZigBee 3.0 多節點組網實戰
- 基于ZigBee的溫濕度 & 信號強度探測系統
- 課外篇:進階選修
- 《課外篇:進階選修》的說明
- 第1章:串口通信協議設計
- 1.1 設計基礎
- 1.2 協議格式
- 第2章:優化協調器工程結構
- 2.1 工程結構
- 2.2 應用框架詳解
- 2.2.1 框架說明
- 2.2.2 zbmsg
- 2.2.3 zbcategory
- 第3章:協調器上位機調試
- 3.1上位機說明
- 3.2 調試說明
- 第4章:信道及PanId的動態修改
- 4.1 串口協議
- 4.2 重要接口說明
- 4.2.1 NIB
- 4.2.2 NLME_UpdateNV
- 4.3 架構調整
- 4.4 應用
- 4.4.1 zbnwk接口實現
- 4.4.2 串口通信解析
- 4.4.3 燒錄調試
- 第5章:網絡短地址及MAC地址的獲取
- 5.1 接口說明
- 5.1.1 描述
- 5.1.2 調用流程
- 5.1.3 異步數據
- 5.2 調試
- 第6章:入網控制及白名單
- 6.1 基本內容
- 6.1.1 入網控制
- 6.1.2 白名單
- 6.2 函數封裝
- 6.3 程序調試
- 第7章:協調器分區存儲管理
- 7.1 軟件框架
- 7.2 應用
- 7.3 調試
- ZigBee 2 WiFi —— 基于ESP8266
- 1.使用云端服務器
- 2.源碼說明與測試
- 3.ESP8266模塊參考資料
- ZigBee 無線報文的抓取與分析
- 接入小米Aqara智能插座和溫濕度傳感器
- Z-Stack的NV應用
- 1. NV 簡介
- 2. NV的讀寫
- 基于HAL的外部FLASH應用
- TFT顯示器實驗(選修)
- Lighting工程源碼分析
- 9.1 ZHA Lighting工程
- 9.2 ZHA Lighting源碼分析
- 9.3 Lighting亮度調節實驗
- TemperatureSensor工程源碼分析
- 10.1 ZHA TemperatureSensor工程
- 10.2 ZHA TemperatureSensor源碼分析
- 版權聲明與免責聲明