当前位置: 首页 > news >正文

Flutter局域网广播(UDP通信)与TCP通信

前言

现在有一个需求,手机和ESP32通过WIFI进行通信。流程如下:

  • 手机创建TCP服务器
  • 手机向192.168.0.255的1002端口广播自己的ip地址以及TCP服务器的端口号
  • ESP32监听到1002的广播内容后,连接手机的TCP服务器。
  • 最后就是ESP32硬件和TCP服务器进行数据收发

因此我们要了解Flutter如何使用UDP进行广播、如何创建TCP服务器以及通过TCP进行通信

验证工具

假如我们代码写好了,和硬件直接调试发现出现了错误,我们就无法定位是Flutter的代码问题还是硬件的问题。因此需要一个验证工具来验证是哪方面出现了问题

这里我们使用验证工具是“山外多功能调试助手”

模拟TCP服务器

  • 打开山外多功能调试助手
  • 点击网络调试助手,再点击TCP服务器

  • 输入端口号点击监听,硬件根据ip地址和端口号进行连接,在消息区我们可以看到是否有设备连接

  • ESP32能够连接这个软件说明错误问题在手机程序中

模拟TCP客户端

  • 前端创建好了TCP服务端后,可以通过这个软件进行连接测试
  • 打开“山外多功能调试助手”,点击网络调试助手,点击TCP客户端
  • 输入服务器IP地址以及端口号,点击连接后查看消息是否连接成功

如果能够连接成功则说明我们的TCP服务器创建成功

模拟UDP广播

  • 点击网络调试助手——>点击UDP,输入广播的端口号后点击连接
  • 手机程序发送一个广播,在消息区可以看到广播的内容

创建TCP服务器

要在Flutter中创建TCP服务器,您可以使用dart:io库中的ServerSocket类。

下面代码使用ServerSocket.bind()方法绑定服务器的地址和端口。然后,我们使用server.listen()来监听来自客户端的连接。当客户端连接到服务器时,我们打印连接信息并向客户端发送欢迎消息。

接下来,我们通过socket.listen()来监听客户端发送的数据。当接收到数据时,我们打印收到的消息,并向客户端发送回复消息。如果在接收数据时出现错误或客户端断开连接,我们关闭与客户端的连接。

startServer() async {try {_serverSocket?.close();_serverSocket = await ServerSocket.bind(InternetAddress.anyIPv4, 5556);print( '服务器已启动,地址:${_serverSocket?.address.address}:${_serverSocket?.port}');_serverSocket?.listen(serverOnReceive);} catch (e, stackTrace) {print('e---------:$e');}
}
serverOnReceive(Socket socket) {_socket = socket;print('客户端已连接: ${socket.remoteAddress}:${socket.remotePort}');socket.writeln('欢迎连接到服务器'); // 向客户端发送欢迎消息_socket?.listen((data) {print('接收到客户端消息: $data');socket.writeln('服务器收到消息: $data'); // 回复客户端收到的消息},onError: (error) {print('接收数据时出现错误: $error');socket.close(); // 关闭与客户端的连接},onDone: () {print('客户端已断开连接');socket.close(); // 关闭与客户端的连接},);
}

局域网广播

权限设置

android

在Flutter中发送广播消息所涉及的权限取决于您的目标平台和网络环境。以下是一些常见的权限,您可能需要在Flutter应用程序中配置以发送广播消息:

对于Android平台:

  • android.permission.INTERNET:用于访问互联网。
  • android.permission.ACCESS_NETWORK_STATE:用于访问网络状态信息。
  • android.permission.ACCESS_WIFI_STATE:用于访问Wi-Fi状态信息。
  • android.permission.CHANGE_WIFI_MULTICAST_STATE:用于更改Wi-Fi多播状态。
对于ios平台
  • NSLocalNetworkUsageDescription:描述应用程序使用本地网络的目的,添加该键值对到应用程序的Info.plist文件中。

示例Info.plist文件中的键值对配置部分:

<key>NSLocalNetworkUsageDescription</key><string>需要使用本地网络以发送广播消息。</string>
具体代码

在Flutter中,可以使用dart:io库中的RawDatagramSocket类来发送广播消息。以下是一个简单的示例代码,演示如何在Flutter中发送广播消息:

void sendBroadcastMessage() {String message = 'BB192.168.0.174:5556';print(utf8.encode(message));RawDatagramSocket.bind(InternetAddress.anyIPv4, 0).then((RawDatagramSocket socket) {rawDatagramSocket = socket;var broadcastAddr = InternetAddress('255.255.255.255');var port = 1002; // 广播目标端口socket.broadcastEnabled = true;socket.send(utf8.encode(message), broadcastAddr, port);socket.close();}).catchError((error) {print('发送广播消息时出现错误: $error');});
}

全部代码

import 'dart:convert';
import 'dart:io';import 'package:flutter/material.dart';ServerSocket? _serverSocket;
Socket? _socket;
RawDatagramSocket? rawDatagramSocket;class TCPServe extends StatefulWidget {const TCPServe({super.key});@overrideState<TCPServe> createState() => _TCPServeState();
}class _TCPServeState extends State<TCPServe> {startServer() async {print("InternetAddress.anyIPv4:${InternetAddress.loopbackIPv4}");try {_serverSocket?.close();// _serverSocket = await ServerSocket.bind("localhost", 1234);_serverSocket = await ServerSocket.bind(InternetAddress.anyIPv4, 5556);print('服务器已启动,地址: ${_serverSocket?.address.address}:${_serverSocket?.port}');_serverSocket?.listen(serverOnReceive);} catch (e, stackTrace) {print('e---------:$e');}}serverOnReceive(Socket socket) {_socket = socket;print('客户端已连接: ${socket.remoteAddress}:${socket.remotePort}');socket.writeln('欢迎连接到服务器'); // 向客户端发送欢迎消息_socket?.listen((data) {print('接收到客户端消息: $data');socket.writeln('服务器收到消息: $data'); // 回复客户端收到的消息},onError: (error) {print('接收数据时出现错误: $error');socket.close(); // 关闭与客户端的连接},onDone: () {print('客户端已断开连接');socket.close(); // 关闭与客户端的连接},);}clean() async {await _socket?.close();print("_socket关闭");await _serverSocket?.close();print("_serverSocket关闭");}void sendBroadcastMessage() {String message = 'BB192.168.0.174:5556';print(utf8.encode(message));RawDatagramSocket.bind(InternetAddress.anyIPv4, 0).then((RawDatagramSocket socket) {rawDatagramSocket = socket;var broadcastAddr = InternetAddress('255.255.255.255');var port = 1002; // 广播目标端口socket.broadcastEnabled = true;socket.send(utf8.encode(message), broadcastAddr, port);socket.close();}).catchError((error) {print('发送广播消息时出现错误: $error');});}cleanRawDatagramSocket() {rawDatagramSocket?.close();}@overrideWidget build(BuildContext context) {return Scaffold(appBar: AppBar(title: const Text("TCP服务端测试")),body: Center(child: Column(children: [TextButton(onPressed: startServer,child: const Text("创建服务"),),TextButton(onPressed: clean,child: const Text("关闭服务"),),TextButton(onPressed: () {_socket?.writeln('REVERSE_LED');},child: const Text("发送REVERSE_LED"),),TextButton(onPressed: sendBroadcastMessage,child: const Text("开始广播"),),TextButton(onPressed: cleanRawDatagramSocket,child: const Text("关闭广播"),),],),),);}
}

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • kafka 消息位移提交几种方式:消息重复消息、消息丢失的关键
  • C++ | Leetcode C++题解之第415题字符串相加
  • Go-知识-定时器
  • KTH5762系列 低功耗、高精度 3D 霍尔角度传感器 电子手表旋钮应用
  • 【对比学习串烧】 SimSiam MoCov3 DINO
  • nacos和eureka的区别
  • java(3)数组的定义与使用
  • 数值实验作业(第一章)
  • 鸿蒙OpenHarmony【轻量系统内核扩展组件(动态加载)】子系统开发
  • Python青少年简明教程目录
  • 混合开发应用侧-JSBridge,在加载的网页中调用原生能力
  • ARM驱动学习之7 驱动模块传参数
  • CentOS入门宝典:从零到一构建你的Linux服务器帝国
  • 64. 求 1+2+…+n
  • MOS工作的三种状态及其分析——亚阈值区(截至区),深三极管区(又叫深线性区)和饱和区
  • 【刷算法】求1+2+3+...+n
  • 230. Kth Smallest Element in a BST
  • Angular2开发踩坑系列-生产环境编译
  • Angular数据绑定机制
  • es6
  • iOS筛选菜单、分段选择器、导航栏、悬浮窗、转场动画、启动视频等源码
  • Linux后台研发超实用命令总结
  • spring + angular 实现导出excel
  • Spring Boot快速入门(一):Hello Spring Boot
  • Travix是如何部署应用程序到Kubernetes上的
  • VUE es6技巧写法(持续更新中~~~)
  • windows下mongoDB的环境配置
  • Yii源码解读-服务定位器(Service Locator)
  • 工作中总结前端开发流程--vue项目
  • 基于axios的vue插件,让http请求更简单
  • 力扣(LeetCode)965
  • 面试遇到的一些题
  • 如何优雅地使用 Sublime Text
  • 微信端页面使用-webkit-box和绝对定位时,元素上移的问题
  • 微信开放平台全网发布【失败】的几点排查方法
  • 仓管云——企业云erp功能有哪些?
  • ​软考-高级-系统架构设计师教程(清华第2版)【第1章-绪论-思维导图】​
  • ​业务双活的数据切换思路设计(下)
  • # centos7下FFmpeg环境部署记录
  • #HarmonyOS:Web组件的使用
  • #我与Java虚拟机的故事#连载09:面试大厂逃不过的JVM
  • (13)Latex:基于ΤΕΧ的自动排版系统——写论文必备
  • (3)Dubbo启动时qos-server can not bind localhost22222错误解决
  • (3)医疗图像处理:MRI磁共振成像-快速采集--(杨正汉)
  • (7) cmake 编译C++程序(二)
  • (Matlab)基于蝙蝠算法实现电力系统经济调度
  • (附源码)计算机毕业设计ssm基于B_S的汽车售后服务管理系统
  • (四)汇编语言——简单程序
  • (已解决)Bootstrap精美弹出框模态框modal,实现js向modal传递数据
  • **登录+JWT+异常处理+拦截器+ThreadLocal-开发思想与代码实现**
  • .NET Core IdentityServer4实战-开篇介绍与规划
  • .net core Swagger 过滤部分Api
  • .net core控制台应用程序初识
  • .Net MVC4 上传大文件,并保存表单
  • .NET 使用配置文件