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

人工智能:扩散模型(Diffusion Model)原理与图像生成实战

人工智能:扩散模型(Diffusion Model)原理与图像生成实战

人工智能:扩散模型(Diffusion Model)原理与图像生成实战 1.1 本章学习目标与重点 💡 学习目标:掌握扩散模型的核心原理、前向扩散与反向扩散过程,以及基于扩散模型的图像生成任务实战流程。 💡 学习重点:理解扩散模型的噪声添加与噪声消除机制,学会使用 PyTorch 搭建 DDPM 模型,完成手写数字图像生成任务。 1.2 扩散模型的核心思想 1.2.1 为什么需要扩散模型 💡 传统的生成模型(如 GAN)存在训练不稳定、模式崩溃等问题。扩散模型作为一种基于概率的生成模型,通过逐步添加噪声和逐步去除噪声的双向过程,实现了更稳定的训练和更高质量的生成效果。 扩散模型的灵感来源于非平衡热力学,它的核心是将复杂的生成问题拆解为多个简单的马尔可夫链步骤。在图像生成、文本生成、语音合成等领域,扩散模型的表现已经超越了传统生成模型。 1.2.2 扩散模型的基本框架 💡 扩散模型包含两个核心过程:前向扩散过程和反向扩散过程。 1. 前向扩散过程:从真实数据出发,

By Ne0inhk
告别重复劳动:用AI数据标注工具提速3倍的实战经验

告别重复劳动:用AI数据标注工具提速3倍的实战经验

👋 大家好,欢迎来到我的技术博客! 📚 在这里,我会分享学习笔记、实战经验与技术思考,力求用简单的方式讲清楚复杂的问题。 🎯 本文将围绕AI这个话题展开,希望能为你带来一些启发或实用的参考。 🌱 无论你是刚入门的新手,还是正在进阶的开发者,希望你都能有所收获! 文章目录 * 告别重复劳动:用AI数据标注工具提速3倍的实战经验 * 为什么数据标注是“效率黑洞”? * AI标注工具的核心优势:不只是快,更是智能 * 实战经验:从0到1的AI标注落地 * 项目背景:一个真实的数据标注挑战 * 工具集成:代码示例详解 * 步骤1:安装依赖库 * 步骤2:加载预训练模型(使用PyTorch) * 步骤3:集成到Label Studio工作流 * 步骤4:人工审核界面优化 * 速度与质量实测数据 * 流程优化:用Mermaid重构标注工作流 * 避坑指南:实战中的常见陷阱 * 陷阱1:AI模型不匹配业务场景 * 陷阱2:数据格式不兼容

By Ne0inhk
被问爆的Agent实战:从0到1搭建可落地AI智能体

被问爆的Agent实战:从0到1搭建可落地AI智能体

🎁个人主页:User_芊芊君子 🎉欢迎大家点赞👍评论📝收藏⭐文章 🔍系列专栏:AI 文章目录: * 【前言】 * 一、先搞懂:2026年爆火的AI Agent,到底是什么? * 1.1 Agent的核心定义 * 1.2 Agent的4大核心能力 * 1.3 2026年Agent的3个热门落地场景 * 二、框架选型:2026年6大主流Agent框架,新手该怎么选? * 三、实战环节:从0到1搭建可落地的“邮件处理Agent”(全程代码+步骤) * 3.1 实战准备:环境搭建(10分钟搞定) * 3.1.1 安装Python环境 * 3.1.2 创建虚拟环境(避免依赖冲突) * 3.

By Ne0inhk
OpenClaw 浏览器控制终极方案 - 让 AI 助手随时控制你的浏览器:

OpenClaw 浏览器控制终极方案 - 让 AI 助手随时控制你的浏览器:

🚀 懒人版:你可能都不用看这篇文章 直接把这篇文章发给 Claude Code,让它帮你执行就行了。它会:创建一个 Chrome Debug 浏览器实例配置好所有参数 然后去 OpenClaw 的 bot 里告诉它: "更新下身份信息:当前你需要去查询信息资料、联网之类的,优先使用已经可以打开的可调试浏览器实例去控制打开搜索等。比如: 使用 --browser-profile mydebug 来控制已打开的浏览器实例。" 搞定!🎉 一个被忽视的痛点 你有没有遇到过这样的场景: 你让 AI 助手帮你搜索信息,它打开了一个全新的浏览器窗口。 然后你发现: * 推特要重新登录 * GitHub 要重新登录 * Google 要重新登录 * 甚至有些网站直接把你当成机器人,拒绝访问 为什么? 因为 AI 助手用的是一个"干净"的浏览器环境,

By Ne0inhk