HarmonyOS基于LYEVK-3861开发火焰报警系统

开发 前端 OpenHarmony
本文内容主要讲述基于LYEVK-3861物联网开发板套件的火焰传感器,开发一个具有火焰感应报警功能的HarmonyOS应用,主要实现蓝牙设备扫描,连接,检测火焰,设置报警阈值。

[[433369]]

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

前言

在各种灾害中,火灾是最经常、最普遍地威胁公众安全和社会发展的主要灾害之一。火给人类带来文明进步、光明和温暖。但是,有时它是人类的朋友,有时是人类的敌人。失去控制的火,就会给人类造成灾难。说到火灾的控制,一套火焰感应报警系统就有其价值了。那我们如何去检测火焰呢?

本文内容主要讲述基于LYEVK-3861物联网开发板套件的火焰传感器,开发一个具有火焰感应报警功能的HarmonyOS应用,主要实现蓝牙设备扫描,连接,检测火焰,设置报警阈值。

1.效果演示

#星光计划1.0# HarmonyOS 基于LYEVK-3861开发火焰报警系统-鸿蒙HarmonyOS技术社区

2.环境准备

本贴不对实验环境搭建做详细说明。具体准备实验环境请参考:

  • 《HarmonyOS 官方文档》
  • LYEVK-3861 物联网开发板套件

3.蓝牙通信说明

3.1 蓝牙通信协议:

3.2 蓝牙通信流程:

#星光计划1.0# HarmonyOS 基于LYEVK-3861开发火焰报警系统-鸿蒙HarmonyOS技术社区

3.3 数据透传协议:

3.3.1 串口协议:

3.3.2 数据协议:

(permission P:APP下发;G:设备请求;R:设备上报)

4.开发调试

蓝牙交互封装BleHelper工具类,通过BLE扫描和广播提供的开放能力,可以根据指定状态获取外围设备、启动或停止BLE扫描、广播。

4.1 进行BLE扫描

  • MyBleCentralManagerCallback继承BleCentralManagerCallback类实现scanResultEvent和scanFailedEvent回调函数,用于接收扫描结果。
  • BleCentralManager(BleCentralManagerCallback callback)接口获取中心设备管理对象。
  • 调用startScan()扫描蓝牙设备。
  1. /** 
  2.    * 扫描设备 
  3.    * @param filters 设备过滤器 
  4.    * @since 2021-10-09 
  5.    */ 
  6.   public void startScan(List<BleScanFilter> filters) { 
  7.       centralManager = new BleCentralManager(context, new MyBleCentralManagerCallback()); 
  8.       centralManager.startScan(filters); 
  9.   } 
  10.  
  11.   /** 
  12.    * 扫描设备回调 
  13.    * 
  14.    * @since 2021-10-09 
  15.    */ 
  16.   private class MyBleCentralManagerCallback implements BleCentralManagerCallback { 
  17.       // 扫描结果的回调 
  18.       @Override 
  19.       public void scanResultEvent(BleScanResult bleScanResult) { 
  20.           if (mBleManagerCallback != null) { 
  21.               mBleManagerCallback.scanResultCallback(bleScanResult); 
  22.           } 
  23.           // 获取广播数据中的服务uuids 
  24.           List<UUID> uuids = bleScanResult.getServiceUuids(); 
  25.           for (UUID uuid : uuids) { 
  26.               if (SERVICE_UUID.equals(uuid.toString())) { 
  27.                   peripheralDevice = bleScanResult.getPeripheralDevice(); 
  28.                   int length = peripheralDevice.toString().length(); 
  29.                   String deviceId = peripheralDevice.toString().substring(length - CUT_LENGTH, length); 
  30.                   stopScan(); 
  31.                   bleConnect(); 
  32.               } 
  33.           } 
  34.       } 
  35.  
  36.       // 扫描失败回调 
  37.       @Override 
  38.       public void scanFailedEvent(int event) { 
  39.           HiLog.debug(loglabel, "扫描失败 scanFailedEvent()"); 
  40.       } 
  41.  
  42.       // 组扫描成功回调 
  43.       @Override 
  44.       public void groupScanResultsEvent(List<BleScanResult> list) { 
  45.           // 使用组扫描时在此对扫描结果进行处理 
  46.       } 
  47.   } 

4.2 建立连接

  • 扫描成功,匹配服务UUID FFB0,调用bleConnect()连接开发板蓝牙。
  • 触发connectionStateChangedEvent(int connectionState)回调,connectionState=2连接成功,然后调用discoverServices()接口发现服务。
  • 在回调servicesDiscoveredEvent(int status)中获取外围设备支持的服务和特征值,此时才能调用read和write方法读取或者写入对应特征值数据。
  • characteristicChangedEvent(GattCharacteristic characteristic)特征变更的回调中,解析传感器上报数据、校验等。具体数据解析逻辑在ProtocolEntity中parseCharacteristic(String hexStr)方法。
  1. /** 
  2.    * 连接到BLE外围设备 
  3.    * 
  4.    * @since 2021-10-09 
  5.    */ 
  6.   public void bleConnect() { 
  7.       peripheralDevice.connect(false, new BlePeripheralCallback() { 
  8.           // 在外围设备上发现服务的回调 
  9.           @Override 
  10.           public void servicesDiscoveredEvent(int status) { 
  11.               super.servicesDiscoveredEvent(status); 
  12.               if (status == BlePeripheralDevice.OPERATION_SUCC) { 
  13.                   HiLog.debug(loglabel, "发现服务成功 servicesDiscoveredEvent()"); 
  14.                   for (GattService service : peripheralDevice.getServices()) { 
  15.                       checkGattCharacteristic(service); 
  16.                   } 
  17.                   if (mBleManagerCallback != null) { 
  18.                       mBleManagerCallback.connectCallback(status); 
  19.                   } 
  20.               } 
  21.           } 
  22.  
  23.           private void checkGattCharacteristic(GattService service) { 
  24.               for (GattCharacteristic tmpChara : service.getCharacteristics()) { 
  25.                   if (tmpChara.getUuid().equals(UUID.fromString(NOTIFY_CHARACTER_UUID))) { 
  26.                       // 启用特征通知 
  27.                       peripheralDevice.setNotifyCharacteristic(tmpChara, true); 
  28.                   } 
  29.                   if (tmpChara.getUuid().equals(UUID.fromString(WRITE_CHARACTER_UUID))) { 
  30.                       // 获取GattCharacteristic 
  31.                       writeCharacteristic = tmpChara; 
  32.                   } 
  33.               } 
  34.           } 
  35.  
  36.           // 连接状态变更的回调 
  37.           @Override 
  38.           public void connectionStateChangeEvent(int connectionState) { 
  39.               super.connectionStateChangeEvent(connectionState); 
  40.               if (connectionState == ProfileBase.STATE_CONNECTED && !isConnected) { 
  41.                   HiLog.debug(loglabel, "连接成功 connectionStateChangeEvent() connectionState:" + connectionState); 
  42.                   isConnected = true
  43.                   // 连接成功在外围设备上发现GATT服务,部分手机发现服务失败,需要延迟调用discoverServices() 
  44.                   peripheralDevice.discoverServices(); 
  45.               } 
  46.           } 
  47.  
  48.           // 特征变更的回调 
  49.           @Override 
  50.           public void characteristicChangedEvent(GattCharacteristic characteristic) { 
  51.               super.characteristicChangedEvent(characteristic); 
  52.               byte[] value = characteristic.getValue(); 
  53.               if (value == nullreturn
  54.               String toHexStr = DataUtils.toHex(value); 
  55.               boolean isVerify = BleHelper.verifyProtocol(toHexStr);//校验 
  56.               if (isVerify) { 
  57.                   if (protocolEntity == null) { 
  58.                       protocolEntity = new ProtocolEntity(); 
  59.                   } 
  60.                   protocolEntity.parseCharacteristic(toHexStr); 
  61.                   if (mBleManagerCallback != null) { 
  62.                       mBleManagerCallback.characteristicChangedCallback(protocolEntity, toHexStr); 
  63.                   } 
  64.               } 
  65.  
  66.           } 
  67.       }); 
  68.   } 

4.3 预警阈值下发

  • 设置火焰报警距离阈值,tagId为0002,当火焰传感器发现火焰,并小于此设置的阈值时,设备上报预警。
  • 通过发现服务servicesDiscoveredEvent()回调获取的writeCharacteristic特征,写入数据。数据下发格式按照3.3数据透传协议。
  1. /** 
  2.      * 主动去获取所Tag设备数据,通过Write(消息类型0x01), Notify接收数据 
  3.      */ 
  4.     public void readInitiativeByTags(List<String> tagIds) { 
  5.         String hex = getTagsCommandHexStr(tagIds); 
  6.         bleManagerWrite(hex); 
  7.     } 
  8.  
  9.     /** 
  10.      * 写Tag设备数据,通过Write(消息类型0x01), Notify接收数据 
  11.      */ 
  12.     public void writeBleByTag(String tagId, String value) { 
  13.         String hex = getTagCommandHexStr(tagId, Integer.parseInt(value)); 
  14.         bleManagerWrite(hex); 
  15.     } 
  16.  
  17.     private void bleManagerWrite(String hex) { 
  18.         if (peripheralDevice == null) { 
  19.             return
  20.         } 
  21.         writeCharacteristic.setValue(hex.getBytes(StandardCharsets.UTF_8)); 
  22.         peripheralDevice.writeCharacteristic(writeCharacteristic); 
  23.     } 
  24.  
  25.     /** 
  26.      * 获取所有Tag数据的Write指令 
  27.      * 
  28.      * @param tagIds tag集 
  29.      */ 
  30.     private String getTagsCommandHexStr(List<String> tagIds) { 
  31.         try { 
  32.             StringBuilder sb = new StringBuilder(); 
  33.             String byte1 = PROTOCOL_ID; //串口协议标识 
  34.             String byte2 = PROTOCOL_VERSION; 
  35.             String byte3 = TYPE_OBTAIN; 
  36.             int dataLen = 8 * tagIds.size(); 
  37.             String byte4 = DataUtils.encodeHex(dataLen, 4); 
  38.  
  39.             String byteTagLen = DataUtils.encodeHex(4, 4); 
  40.             String byteTagValue = "00000000"
  41.             StringBuilder sbTag = new StringBuilder(); 
  42.             for (String it : tagIds) { 
  43.                 sbTag.append(it).append(byteTagLen).append(byteTagValue); 
  44.             } 
  45.  
  46.             sb.append(byte1).append(byte2).append(byte3).append(byte4).append(sbTag); 
  47.             String hexStrSum = DataUtils.makeChecksum(sb.toString()); 
  48.             int protocolVerify = DataUtils.decodeHEX(hexStrSum) % 256; 
  49.             String byteLast = DataUtils.encodeHex(protocolVerify); 
  50.             sb.append(byteLast); 
  51.             return sb.toString(); 
  52.         } catch (Exception e) { 
  53.             e.printStackTrace(); 
  54.         } 
  55.         return ""
  56.     } 
  57.  
  58.     /** 
  59.      * 获取单个Tag数据的Write指令 
  60.      * 
  61.      * @param tagId  0001 
  62.      * @Param value  写入值 
  63.      */ 
  64.     private String getTagCommandHexStr(String tagId, int value) { 
  65.         try { 
  66.             StringBuilder sb = new StringBuilder(); 
  67.             String byte1 = PROTOCOL_ID; //串口协议标识 
  68.             String byte2 = PROTOCOL_VERSION; 
  69.             String byte3 = TYPE_ISSUED; 
  70.             int dataLen = 8; 
  71.             String byte4 = DataUtils.encodeHex(dataLen, 4); 
  72.  
  73.             String byteTagId = tagId; 
  74.             String byteTagLen = DataUtils.encodeHex(4, 4); 
  75.             String byteTagValue = DataUtils.encodeHex(value, 8); 
  76.  
  77.             sb.append(byte1).append(byte2).append(byte3).append(byte4) 
  78.                    .append(byteTagId).append(byteTagLen).append(byteTagValue); 
  79.  
  80.             String hexStrSum = DataUtils.makeChecksum(sb.toString()); 
  81.             int protocolVerify = DataUtils.decodeHEX(hexStrSum) % 256; 
  82.             String byteLast = DataUtils.encodeHex(protocolVerify); 
  83.             sb.append(byteLast); 
  84.             return sb.toString(); 
  85.         } catch (Exception e) { 
  86.             e.printStackTrace(); 
  87.         } 
  88.         return ""
  89.     } 

4.4 火焰距离上报

  • 火焰距离上报,通知应用,开始预警。
  • 通过3.3数据透传协议,解析设备上报的火焰距离。
  1. /** 
  2.   * 协议校验 (从协议标识首节至协议内容尾字节求累加和后再对 256 取余) 
  3.   * 
  4.   * @param hexStr 带空格的 A5 5A 01 00 00 08 00 02 00 04 00 00 00 8C 
  5.   */ 
  6.  public static boolean verifyProtocol(String hexStr) { 
  7.      if (hexStr.isEmpty()) return false
  8.      String checkHex = hexStr.substring(0, hexStr.lastIndexOf(" ")); 
  9.      String lastHex = hexStr.substring(hexStr.lastIndexOf(" ") + 1); 
  10.      String hexStrSum = DataUtils.makeChecksum(checkHex); 
  11.      return DataUtils.decodeHEX(hexStrSum) % 256 == DataUtils.decodeHEX(lastHex); 
  12.  } 
  1. /** 
  2.    * 根据串口协议解析 
  3.    */ 
  4.   public boolean parseCharacteristic(String hexStr) { 
  5.       try { 
  6.           String[] hexs = hexStr.split(" "); 
  7.           version = DataUtils.decodeHEX(hexs[2]); 
  8.           messageType = DataUtils.decodeHEX(hexs[3]); 
  9.           dataLen = DataUtils.decodeHEX(hexs[4] + hexs[5]); 
  10.           for (int i = 0; i < dataLen / 8; i++) { 
  11.               int startIndex = 6 + (i * 8); 
  12.               DeviceData deviceData = new DeviceData(); 
  13.               deviceData.tagId = DataUtils.decodeHEX(hexs[startIndex] + hexs[startIndex + 1]); 
  14.               deviceData.len = DataUtils.decodeHEX(hexs[startIndex + 2] + hexs[startIndex + 3]); 
  15.               deviceData.value = DataUtils.decodeHEX(hexs[startIndex + 4] + hexs[startIndex + 5] + 
  16.                       hexs[startIndex + 6] + hexs[startIndex + 7]); 
  17.               deviceListMap.put(deviceData.tagId, deviceData); 
  18.           } 
  19.           protocolVerify = DataUtils.decodeHEX(hexs[6 + dataLen]); 
  20.       } catch (Exception e) { 
  21.           e.printStackTrace(); 
  22.           return false
  23.       } 
  24.       return true
  25.   } 

5.结语

以上就是LYEVK-3861物联网开发板火焰传感器的预警功能,和应用程序交互的一个相对简单的流程。场景的交互还有很多种,比如在此基础上搭建远程云端系统,实现远程火焰监控实时预警。有兴趣的伙伴也可以根据开发板其他传感器组合成不同的智能场景。

想了解更多内容,请访问:

51CTO和华为官方合作共建的鸿蒙技术社区

https://harmonyos.51cto.com

 

责任编辑:jianghua 来源: 鸿蒙社区
相关推荐

2021-08-27 17:02:56

鸿蒙HarmonyOS应用

2021-10-11 10:07:09

鸿蒙HarmonyOS应用

2021-09-02 15:25:53

鸿蒙HarmonyOS应用

2021-08-17 14:40:43

鸿蒙HarmonyOS应用

2021-09-16 10:03:39

鸿蒙HarmonyOS应用

2009-11-20 09:17:20

Firefox操作系统计划

2020-12-15 11:57:49

Hi3861 HarmonyOS开发板

2020-11-30 13:57:48

Hi3861

2021-09-09 10:06:09

鸿蒙HarmonyOS应用

2022-04-13 11:24:18

ETS开发HarmonyOS鸿蒙

2021-08-18 10:06:33

鸿蒙HarmonyOS应用

2021-12-06 15:05:41

鸿蒙HarmonyOS应用

2022-02-21 15:25:47

HarmonyOS鸿蒙低代码开发

2010-03-24 09:07:17

FirefoxWindows Mob

2021-12-20 21:01:52

鸿蒙HarmonyOS应用

2020-10-23 09:47:19

HarmonyOSJS开发框架

2022-06-28 08:40:16

LokiPromtail日志报警

2020-11-11 11:56:05

HarmonyOS

2022-09-09 14:42:17

应用开发ETS

2014-07-17 15:01:21

zabbix监控开发
点赞
收藏

51CTO技术栈公众号