Flutter蓝牙BLE开发完全指南(内含高级功能扩展)

Flutter蓝牙BLE开发完全指南

我将为您提供一个完整的Flutter蓝牙BLE实现方案,包含UI设计、权限处理、设备扫描、连接通信等完整功能。

完整实现方案

1. 添加依赖与权限配置

pubspec.yaml

dependencies:flutter:sdk: flutterflutter_blue_plus: ^1.10.0permission_handler: ^10.4.0provider: ^6.0.5animated_text_kit: ^4.2.2percent_indicator: ^4.2.3

AndroidManifest.xml

<uses-permission android:name="android.permission.BLUETOOTH"/>
<uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION"/>
<uses-permission android:name="android.permission.BLUETOOTH_CONNECT"/>
<uses-permission android:name="android.permission.BLUETOOTH_SCAN"/>
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION"/>
<uses-permission android:name="android.permission.ACCESS_BACKGROUND_LOCATION"/>

Info.plist (iOS)

<key>NSBluetoothAlwaysUsageDescription</key>
<string>需要蓝牙权限来连接设备</string>
<key>NSBluetoothPeripheralUsageDescription</key>
<string>需要蓝牙权限来连接设备</string>
<key>UIBackgroundModes</key>
<array><string>bluetooth-central</string><string>bluetooth-peripheral</string>
</array>

2. 蓝牙状态管理类

import 'package:flutter_blue_plus/flutter_blue_plus.dart';
import 'package:provider/provider.dart';class BluetoothProvider extends ChangeNotifier {List<ScanResult> scannedDevices = [];BluetoothDevice? connectedDevice;List<BluetoothService> services = [];BluetoothCharacteristic? targetCharacteristic;bool isScanning = false;bool isConnecting = false;bool isDiscovering = false;String connectionStatus = "未连接";List<List<int>> receivedData = [];Future<void> startScan() async {if (isScanning) return;scannedDevices.clear();isScanning = true;notifyListeners();try {await FlutterBluePlus.startScan(timeout: const Duration(seconds: 15),androidUsesFineLocation: true,);FlutterBluePlus.scanResults.listen((results) {scannedDevices = results;notifyListeners();});} catch (e) {print("扫描错误: $e");isScanning = false;notifyListeners();}}Future<void> stopScan() async {if (!isScanning) return;await FlutterBluePlus.stopScan();isScanning = false;notifyListeners();}Future<void> connectToDevice(BluetoothDevice device) async {if (isConnecting) return;isConnecting = true;connectionStatus = "连接中...";notifyListeners();try {await device.connect(autoConnect: false);connectedDevice = device;connectionStatus = "已连接";// 监听连接状态device.connectionState.listen((state) {if (state == BluetoothConnectionState.disconnected) {connectedDevice = null;services.clear();targetCharacteristic = null;connectionStatus = "已断开";notifyListeners();}});// 发现服务isDiscovering = true;notifyListeners();services = await device.discoverServices();isDiscovering = false;notifyListeners();} catch (e) {print("连接错误: $e");connectionStatus = "连接失败";isConnecting = false;notifyListeners();}isConnecting = false;notifyListeners();}Future<void> disconnectDevice() async {if (connectedDevice == null) return;try {await connectedDevice!.disconnect();} catch (e) {print("断开连接错误: $e");} finally {connectedDevice = null;services.clear();targetCharacteristic = null;connectionStatus = "已断开";notifyListeners();}}Future<void> setCharacteristic(BluetoothCharacteristic characteristic) async {targetCharacteristic = characteristic;// 设置通知监听await characteristic.setNotifyValue(true);characteristic.value.listen((value) {receivedData.add(value);notifyListeners();});notifyListeners();}Future<void> writeData(List<int> data) async {if (targetCharacteristic == null) return;try {await targetCharacteristic!.write(data);} catch (e) {print("写入错误: $e");}}
}

3. 主界面实现

import 'package:flutter/material.dart';
import 'package:flutter_blue_plus/flutter_blue_plus.dart';
import 'package:provider/provider.dart';
import 'bluetooth_provider.dart';void main() {runApp(ChangeNotifierProvider(create: (context) => BluetoothProvider(),child: const MyApp(),),);
}class MyApp extends StatelessWidget {const MyApp({super.key});Widget build(BuildContext context) {return MaterialApp(title: 'Flutter蓝牙BLE',theme: ThemeData(primarySwatch: Colors.blue,scaffoldBackgroundColor: Colors.grey[100],appBarTheme: const AppBarTheme(backgroundColor: Colors.deepPurple,foregroundColor: Colors.white,),),home: const BluetoothHomePage(),);}
}class BluetoothHomePage extends StatefulWidget {const BluetoothHomePage({super.key});State<BluetoothHomePage> createState() => _BluetoothHomePageState();
}class _BluetoothHomePageState extends State<BluetoothHomePage> {void initState() {super.initState();_checkPermissions();}Future<void> _checkPermissions() async {// 检查并请求权限final locationStatus = await Permission.location.request();final bluetoothStatus = await Permission.bluetooth.request();if (locationStatus.isDenied || bluetoothStatus.isDenied) {showDialog(context: context,builder: (context) => AlertDialog(title: const Text('权限不足'),content: const Text('需要位置和蓝牙权限才能扫描设备'),actions: [TextButton(onPressed: () => Navigator.pop(context),child: const Text('确定'),),],),);}}Widget build(BuildContext context) {final bluetoothProvider = Provider.of<BluetoothProvider>(context);return Scaffold(appBar: AppBar(title: const Text('Flutter蓝牙BLE'),actions: [IconButton(icon: const Icon(Icons.bluetooth),onPressed: () {if (bluetoothProvider.connectedDevice != null) {bluetoothProvider.disconnectDevice();} else {Navigator.push(context,MaterialPageRoute(builder: (context) => const DeviceScanPage(),),);}},),],),body: _buildBody(bluetoothProvider),);}Widget _buildBody(BluetoothProvider provider) {if (provider.connectedDevice == null) {return Center(child: Column(mainAxisAlignment: MainAxisAlignment.center,children: [const Icon(Icons.bluetooth_disabled, size: 100, color: Colors.grey),const SizedBox(height: 20),const Text('未连接设备',style: TextStyle(fontSize: 24, fontWeight: FontWeight.bold),),const SizedBox(height: 10),ElevatedButton(onPressed: () {Navigator.push(context,MaterialPageRoute(builder: (context) => const DeviceScanPage(),),);},child: const Text('扫描设备'),),],),);}return DefaultTabController(length: 3,child: Column(children: [_buildDeviceHeader(provider),const TabBar(tabs: [Tab(icon: Icon(Icons.devices), text: '设备信息'),Tab(icon: Icon(Icons.settings), text: '服务'),Tab(icon: Icon(Icons.message), text: '通信'),],),Expanded(child: TabBarView(children: [_buildDeviceInfo(provider),_buildServices(provider),_buildCommunication(provider),],),),],),);}Widget _buildDeviceHeader(BluetoothProvider provider) {return Container(padding: const EdgeInsets.all(16),color: Colors.deepPurple[50],child: Row(children: [const Icon(Icons.bluetooth_connected, color: Colors.deepPurple, size: 36),const SizedBox(width: 16),Expanded(child: Column(crossA

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/diannao/93376.shtml
繁体地址,请注明出处:http://hk.pswp.cn/diannao/93376.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用 Canvas 替代 <video> 标签加载并渲染视频

在部分浏览器环境或业务场景下&#xff0c;直接使用 <video> 标签加载视频会出现首帧延迟的情况。以下方法通过 WebGPU Canvas 2D 将视频帧绘制到自定义 Canvas 上&#xff0c;让 <video> 只做解码&#xff0c;WebGPU 接管渲染&#xff0c;通过最小化对象创建 精…

基于Flask的智能停车场管理系统开发实践

在现代城市中&#xff0c;停车难已成为一个普遍问题。为了解决这一问题&#xff0c;我开发了一个基于Python Flask框架的智能停车场管理系统。该系统集成了车牌识别、车位状态监控、收费管理等多项功能&#xff0c;为停车场的智能化管理提供了完整的解决方案。系统功能概述该停…

【C#获取高精度时间】

在C#中&#xff0c;有几种方法可以获取高精度时间&#xff08;高分辨率时间戳&#xff09;&#xff0c;适用于性能测量、计时等需要高精度的场景。以下是几种常用方法&#xff1a; 1. 使用 Stopwatch 类&#xff08;推荐&#xff09; Stopwatch 类提供了最高精度的时间测量&…

Spring Boot + React 打造现代化高校成绩管理系统实战记录

作者: 笙囧同学 发布时间: 2025年7月 技术栈: Spring Boot 3.2.3 React 18 TypeScript 华为云GaussDB 项目类型: 全栈Web应用 开发周期: 30天 代码量: 15000 行 &#x1f4d6; 前言 大家好&#xff0c;我是笙囧同学&#xff01;&#x1f64b;‍♂️ 作为一名计算机科学与技…

形参表不匹配(BUG)

在您的代码中&#xff0c;存在两个主要问题导致"形参表中不匹配"的错误&#xff1a;erase() 函数中的成员变量名错误iterator erase(iterator pos) {// ...size--; // ❌ 错误&#xff1a;成员变量名为 _size 而非 sizereturn iterator(next); }修正&#xff1a;ite…

Spring循环依赖以及三个级别缓存

Spring循环依赖以及三个级别缓存 什么是循环依赖&#xff1f; 循环依赖&#xff0c;顾名思义&#xff0c;就是指两个或多个 Spring Bean 之间相互依赖&#xff0c;形成一个闭环。 最常见也是 Spring 能够“解决”的循环依赖是构造器注入 和 setter 注入 混合或单独使用时&…

《零基础入门AI:OpenCV图像预处理进一步学习》

本文全面讲解OpenCV图像预处理的七大核心技术&#xff08;插值方法、边缘填充、图像矫正&#xff08;透视变换&#xff09;、图像掩膜、ROI切割、图像添加水印、图像噪点消除&#xff09;&#xff0c;每个知识点都配有详细解释和实用代码示例&#xff0c;帮助初学者建立系统的图…

MongoDB的内存和核心数对于运行效率的影响

在 MongoDB 线上生产环境中&#xff0c;CPU&#xff08;核心&#xff09; 和 内存 是两大关键硬件资源&#xff0c;它们在不同的操作场景下发挥着核心作用&#xff0c;共同影响着数据库的性能、稳定性和扩展性。理解它们的作用场景至关重要&#xff0c;是容量规划、性能优化和故…

自己的SAPGUI尝试

为满足用户需求&#xff0c;博主做了一个台账管理程序&#xff0c;尝试用自己的程序做GUI&#xff0c;用SAP 系统做数据库。 运行了半年&#xff0c;程序很nice,用户每天都在高效的使用&#xff0c;已经有十几万的数据。 总结一下这次自己的GUI尝试&#xff0c;好处是C# WINFOR…

高效处理 JSON 数据:JsonUtil 工具类全方位解析与实战

在现代软件开发中,JSON(JavaScript Object Notation)已成为数据交换的“通用语言”——从前后端接口通信到微服务数据交互,从配置文件解析到日志格式化,几乎所有场景都离不开JSON的处理。然而,原生JSON框架(如FastJSON、Jackson)的API往往需要大量重复代码,且空指针、…

Python 库手册:xmlrpc.client 与 xmlrpc.server 模块

xmlrpc.client 和 xmlrpc.server 是 Python 标准库中用于构建基于 XML-RPC 协议的远程过程调用&#xff08;RPC&#xff09;通信模块。xmlrpc.client 用于编写客户端程序&#xff0c;向远程服务器发起方法调用。xmlrpc.server 用于编写服务器端&#xff0c;暴露本地方法供远程客…

渲染篇(一):从零实现一个“微型React”:Virtual DOM的真面目

渲染篇(一)&#xff1a;从零实现一个“微型React”&#xff1a;Virtual DOM的真面目 引子&#xff1a;前端性能的“永恒之问” 在前面两章中&#xff0c;我们已经奠定了坚实的架构基础。我们用“任务调度器”建立了声明式和模块化的编程范式&#xff0c;并通过对比MVC等模式论…

SWC 深入全面讲解

一、核心功能与原理 1. 高性能编译 Rust 架构优势&#xff1a;SWC 基于 Rust 编写&#xff0c;利用 Rust 的性能和并发性优势&#xff0c;编译速度比 Babel 快约 20 倍&#xff0c;比 TypeScript 编译器更快。并行编译&#xff1a;支持多线程并行处理&#xff0c;在四核基准测试…

XML Expat Parser:深入解析与高效应用

XML Expat Parser:深入解析与高效应用 引言 XML(可扩展标记语言)作为一种广泛使用的标记语言,在数据交换、存储和表示中扮演着重要角色。XML Expat Parser 是一个高性能、可扩展的XML解析库,广泛应用于各种编程语言中。本文将深入探讨XML Expat Parser 的原理、特性以及…

【Python】自动化GIT提交

在日常开发中&#xff0c;我们经常需要频繁地向 Git 仓库提交代码。虽然 git add、git commit、git push 这几个命令并不复杂&#xff0c;但重复操作容易出错&#xff0c;也浪费时间。本文将介绍如何使用 Python 脚本自动化完成 Git 提交流程&#xff0c;让开发更高效&#xff…

基于Qlearning强化学习的水下无人航行器路径规划与避障系统matlab性能仿真

目录 1.引言 2.算法仿真效果演示 3.数据集格式或算法参数简介 4.算法涉及理论知识概要 5.参考文献 6.完整算法代码文件获得 1.引言 水下无人航行器 (Autonomous Underwater Vehicle, AUV) 的路径规划与避障是海洋探索、资源开发和军事应用中的关键技术。传统的路径规划方…

模块自由拼装!Python重构DSSAT作物模块教程(以杂交水稻为例)

基于过程的作物生长模型&#xff08;Process-based Crop Growth Simulation Model&#xff09;在模拟作物对气候变化的响应与适应、农田管理优化、作物品种和株型筛选、农业碳中和、农田固碳减排等领域扮演着越来越重要的作用。Decision Support Systems for Agrotechnology Tr…

Java项目接口权限校验的灵活实现

引言 在Java Web开发中&#xff0c;接口权限校验是保护系统资源安全的关键机制。本文将介绍一种灵活、可配置的接口权限校验方案&#xff0c;通过注解驱动和拦截器实现&#xff0c;既能保证安全性&#xff0c;又能灵活控制哪些接口需要校验。 设计思路 实现方案的核心设计要点&…

瀚高DB兼容MySQL if函数

文章目录环境症状问题原因解决方案环境 系统平台&#xff1a;Linux x86-64 Red Hat Enterprise Linux 7 版本&#xff1a;4.5 症状 MySQL if函数在瀚高DB当中没有&#xff0c;源应用在用到if函数时&#xff0c;就会报if函数不存在的错误信息。为此&#xff0c;我们需要根据业…

基于深度学习的胸部 X 光图像肺炎分类系统(六)

目录 结果指标解读 一、为什么选择这些指标&#xff1f; 二、各指标的定义和解读 1. 准确率&#xff08;Accuracy&#xff09; 2. 损失&#xff08;Loss&#xff09; 3. 精确率&#xff08;Precision&#xff09; 4. 召回率&#xff08;Recall&#xff09; 三、这些指标…