Flutter for OpenHarmony:multicast_dns 发现局域网设备,实现零配置网络 (mDNS/Bonjour/Zeroconf)(本地服务发现) 深度解析与鸿蒙适配指南

Flutter for OpenHarmony:multicast_dns 发现局域网设备,实现零配置网络 (mDNS/Bonjour/Zeroconf)(本地服务发现) 深度解析与鸿蒙适配指南

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net

在这里插入图片描述

前言

在智能家居、打印机共享、P2P 文件传输等场景中,设备往往通过局域网互联。但 IP 地址是动态分配的,用户不可能去记 192.168.1.105。mDNS(Multicast DNS)允许设备在没有 DNS 服务器的情况下,通过 .local 域名互相发现。

multicast_dns 是 Dart 官方提供的库,用于发送和接收 mDNS 报文。你可以用它来查找局域网内的 HTTP 服务、Google Cast 设备,或者广播自己的服务供他人发现。

一、概念介绍/原理解析

1.1 基础概念

  • Multicast (组播): 向特定组播地址(224.0.0.251)发送数据包,局域网所有设备都能收到。
  • Service Type (服务类型): 如 _http._tcp (Web 服务), _googlecast._tcp (Chromecast)。
  • Record (记录): SRV (端口), TXT (元数据), A (IP)。

查询: _http._tcp.local

响应: 佳能打印机 @ 192.168.1.5

响应: 客厅电视 @ 192.168.1.8

手机终端

局域网组播

打印机

电视机

1.2 进阶概念

虽然 iOS/Android/macOS 系统层面都有 mDNS 服务(Bonjour/NsdManager),但 multicast_dns 是在 Dart 层直接操作 UDP Socket 实现的,因此跨平台一致性较好,但也受制于 Socket 权限。

二、核心 API/组件详解

2.1 基础用法

搜索局域网内的 Google Cast 设备。

import'package:multicast_dns/multicast_dns.dart';voidmain()async{final client =MDnsClient();await client.start();// 查询指针记录 (PTR)awaitfor(final ptr in client.lookup<PtrResourceRecord>(ResourceRecordQuery.serverPointer('_googlecast._tcp.local'),)){print('发现设备: ${ptr.domainName}');// 进一步查询详情 (SRV, IP)awaitfor(final srv in client.lookup<SrvResourceRecord>(ResourceRecordQuery.service(ptr.domainName),)){print('端口: ${srv.port}, 主机: ${srv.target}');}} client.stop();}
在这里插入图片描述

2.2 广播服务

目前库主要侧重于 客户端查询。要想作为服务端广播,通常需要更底层的 API 或者依赖平台插件(如 nsd)。但若是纯 Dart 环境,可以手动构造响应包(较复杂)。

三、常见应用场景

3.1 场景 1:查找打印机

搜索 _ipp._tcp_printer._tcp 服务。

awaitfor(final ptr in client.lookup<PtrResourceRecord>(ResourceRecordQuery.serverPointer('_ipp._tcp.local'),)){print('Found Printer: ${ptr.domainName}');}
在这里插入图片描述

3.2 场景 2:局域网对战游戏

两台手机进入同一 WiFi,互相发现对方建立连接。

// 搜索自定义的游戏服务类型final query =ResourceRecordQuery.serverPointer('_mygame._udp.local');awaitfor(final ptr in client.lookup<PtrResourceRecord>(query)){// 连接到发现的对战房间_connectToRoom(ptr.domainName);}
在这里插入图片描述

3.3 场景 3:开发调试

在 PC 开启服务,手机自动寻找调试端口,免去手动输入 IP。

// PC端可能通过 avahi 广播了 _debug._tcpfinal srv =await client.lookup<SrvResourceRecord>(ResourceRecordQuery.service('pc-debug._debug._tcp.local'),).first;print('Connect to Debugger: ${srv.target}:${srv.port}');
在这里插入图片描述

四、OpenHarmony 平台适配

4.1 组播权限与锁

在移动设备上监听组播需要特殊权限,并且在某些省电模式下,系统会过滤组播包。在 OpenHarmony 上,记得申请 INTERNET 权限。

4.2 网络绑定

多网卡设备(如同时开启 WiFi 和热点)可能导致 mDNS 发到了错误的网卡。MDnsClient 允许指定 RawDatagramSocket.bind 的参数。如果您发现搜不到设备,尝试检查网络接口。

五、完整示例代码

本示例构建一个局域网设备扫描器,列出所有 HTTP 服务 (_http._tcp)。

import'package:flutter/material.dart';import'package:multicast_dns/multicast_dns.dart';voidmain(){runApp(constMaterialApp(home:MDnsPage()));}classMDnsPageextendsStatefulWidget{constMDnsPage({super.key});@overrideState<MDnsPage>createState()=>_MDnsPageState();}classServiceInfo{finalString name;finalString host;final int port;finalString ip;ServiceInfo(this.name,this.host,this.port,this.ip);}class _MDnsPageState extendsState<MDnsPage>{finalList<ServiceInfo> _services =[]; bool _scanning =false;MDnsClient? _client;Future<void>_startScan()async{setState((){ _services.clear(); _scanning =true;}); _client =MDnsClient();await _client!.start();try{// 1. 查找服务实例 (PTR)awaitfor(final ptr in _client!.lookup<PtrResourceRecord>(ResourceRecordQuery.serverPointer('_http._tcp.local'),)){// 2. 查找服务详情 (SRV)awaitfor(final srv in _client!.lookup<SrvResourceRecord>(ResourceRecordQuery.service(ptr.domainName),)){// 3. 查找 IP (A)awaitfor(final ip in _client!.lookup<IPAddressResourceRecord>(ResourceRecordQuery.addressIPv4(srv.target),)){final info =ServiceInfo( ptr.domainName, srv.target, srv.port, ip.address.address,);if(mounted){setState((){if(!_services.any((s)=> s.ip == info.ip && s.port == info.port)){ _services.add(info);}});}}}}}finally{ _client?.stop();if(mounted)setState(()=> _scanning =false);}}@overridevoiddispose(){ _client?.stop();super.dispose();}@overrideWidgetbuild(BuildContext context){returnScaffold( appBar:AppBar(title:constText('局域网 HTTP 服务发现')), body:Column( children:[if(_scanning)constLinearProgressIndicator(),Expanded( child: _services.isEmpty ?constCenter(child:Text('没有发现 HTTP 服务,请确保 WiFi 连接')):ListView.builder( itemCount: _services.length, itemBuilder:(context, index){final s = _services[index];returnListTile( leading:constIcon(Icons.computer), title:Text(s.name.replaceAll('._http._tcp.local','')), subtitle:Text('${s.host}:${s.port} (${s.ip})'),);},),),],), floatingActionButton:FloatingActionButton( onPressed: _scanning ?null: _startScan, child:constIcon(Icons.refresh),),);}}
在这里插入图片描述

六、总结

multicast_dns 是 Dart 原生的 mDNS 解决方案。虽然不如原生 API 稳定(受限于 UDP 组播环境),但其跨平台一致性和灵活性是巨大的优势。

最佳实践

  1. 超时:mDNS 查询可能永远挂起(如果没有响应),务必结合 timeout 或手动 stop
  2. 缓存:设备名称与 IP 的映射关系应有短时缓存,减少网络风暴。
  3. 兼容性:Android/OpenHarmony 某些版本默认禁止后台组播,确保 App 前台运行且权限正确。

Read more

Flutter 三方库 sm_crypto 的鸿蒙化适配指南 - 实现国产密码算法 SM2/SM3/SM4 的端侧加解密、支持数字签名与国密 SSL 安全通信实战

Flutter 三方库 sm_crypto 的鸿蒙化适配指南 - 实现国产密码算法 SM2/SM3/SM4 的端侧加解密、支持数字签名与国密 SSL 安全通信实战

欢迎加入开源鸿蒙跨平台社区:https://openharmonycrossplatform.ZEEKLOG.net Flutter 三方库 sm_crypto 的鸿蒙化适配指南 - 实现国产密码算法 SM2/SM3/SM4 的端侧加解密、支持数字签名与国密 SSL 安全通信实战 前言 在进行针对中国市场的 Flutter for OpenHarmony 企业级或政务级应用开发时,支持国产密码算法(国密)是硬性的合规要求。sm_crypto 是一个功能完备的国密算法 Dart 实现库。它涵盖了非对称加密 SM2、哈希摘要 SM3 以及对称加密 SM4。本文将探讨如何在鸿蒙端利用该库构建符合国家标准的安全加密体系。 一、原原理性解析 / 概念介绍 1.1 基础原理 sm_crypto 严格遵循国家密码管理局发布的 GM/

By Ne0inhk
【算法通关指南:算法基础篇】二分答案专题:1.木材加工 2.砍树

【算法通关指南:算法基础篇】二分答案专题:1.木材加工 2.砍树

🔥小龙报:个人主页 🎬作者简介:C++研发,嵌入式,机器人方向学习者 ❄️个人专栏:《算法通关指南 》 ✨ 永远相信美好的事情即将发生 文章目录 * 前言 * 一、二分答案 * 二、二分答案经典算题 * 2.1 木材加工 * 2.1.1题目 * 2.1.2 算法原理 * 2.1.3 代码 * 2.2 砍树 * 2.2.1 题目 * 2.2.2 算法原理 * 2.2.3 代码 * 总结与每日励志 前言 二分答案是算法竞赛与笔试中极具技巧性的高分解法,核心思路是将复杂求解转化为简洁的二分+判定,

By Ne0inhk

Python 集合比列表快得多,对吗?

原文:towardsdatascience.com/python-set-is-way-faster-than-list-true-or-false-042c6f8975cd https://github.com/OpenDocCN/towardsdatascience-blog-zh-2024/raw/master/docs/img/4b79e9e630ef9b8e2b14c7bc20892abe.png 由作者在 Canva 中创建 几周前,我写了一篇另一篇文章来解释一些流行的“Python 小技巧”背后的机制和逻辑。其中之一是在可能的情况下使用 Python 集合而不是列表。 许多文章告诉你 Python 小技巧,但很少告诉你为什么 在这篇文章变得流行之后,许多读者向我提问或争论说 Python 集合并不总是很快。这是绝对正确的。因此,我决定写这篇文章,深入探讨 Python 列表和集合的数据结构。 在这篇文章中,我将首先使用实际代码在不同场景下比较 Python 列表和集合的性能。然后,我将介绍它们使用的几种数据结构,即动态数组和哈希表。基于这些数据结构的特性,我将

By Ne0inhk

Python 爬虫实战:精准抓取携程旅行酒店价格数据

大会官网:https://ais.cn/u/Y3aAzy 会议时间:2026年2月6-8日 会议地点:中国-广州 前言 携程旅行作为国内领先的在线旅游平台,其酒店价格数据包含实时房价、房型信息、优惠活动、用户评分等核心维度,是旅游数据分析、价格监控、竞品分析的重要数据源。相较于博客园的静态页面,携程酒店页面融合了动态加载、反爬验证、数据加密等机制,抓取难度更高。本文将从页面分析、反反爬策略、动态数据抓取等维度,系统讲解如何使用 Python 实现携程酒店价格数据的高效抓取,帮助开发者突破平台限制,获取结构化的酒店价格信息。 摘要 本文聚焦携程旅行酒店价格爬虫的全流程实现,核心涵盖动态页面数据抓取、请求头加密参数处理、分页与多城市数据抓取三大核心技术点,通过requests库发送 HTTP 请求、jsonpath解析 JSON 数据、fake-useragent伪装请求特征,结合实战案例完成酒店名称、价格、房型、评分、位置等核心字段的抓取。实战目标链接:

By Ne0inhk