【Flutter】Dart:Isolate
在 Dart 和 Flutter 中,所有的代码默认都运行在单一的线程(即主线程)上,这个线程也叫做 UI 线程。当进行耗时操作(如复杂计算或网络请求)时,如果不使用多线程处理,主线程会被阻塞,导致应用界面卡顿、无响应,影响用户体验。为了解决这个问题,Dart 提供了 Isolate,一种独立的执行单元,可以并行执行任务。
本教程将深入介绍 Dart 中的 Isolate,涵盖其含义、事件循环、消息传递机制以及如何在不同 Isolate 之间进行通信。
什么是 Isolate
在 Dart 中,Isolate 是一种独立的执行单元,它和线程的概念相似,但与传统的多线程不同。每个 Isolate 都拥有独立的内存堆和事件循环,因此 Isolate 之间不能直接共享内存,而是通过消息传递进行通信。由于隔离的内存管理,Isolate 能够有效避免多线程中的竞态条件和数据竞争问题。
单线程与 Isolate 的区别
-
单线程:Dart 默认在单一线程上运行任务。单线程模型避免了复杂的线程同步问题,但在处理耗时任务时会阻塞主线程,影响应用的响应性。
-
Isolate:Isolate 是独立的执行单元,能够并行处理任务。每个 Isolate 都有自己的内存空间,不与其他 Isolate 共享数据,因此不会出现线程竞争问题。
示例:传统单线程任务阻塞
void main() {print('Start');// 模拟耗时任务for (int i = 0; i < 1000000000; i++) {}print('End');
}
在上述代码中,耗时任务会阻塞主线程,导致应用无法响应用户操作。为了解决这种问题,可以使用 Isolate 来将任务移出主线程。
Isolate 的事件循环与并行执行
每个 Isolate 都有自己的 事件循环,负责管理消息队列并处理异步任务。Dart 中的异步操作(如 Future 和 Stream)也都是通过事件循环来调度的。当一个 Isolate 接收到消息时,它会将消息放入事件队列,并在合适的时机进行处理。
如何创建 Isolate
可以通过 Isolate.spawn() 来创建新的 Isolate。该方法会启动一个新的 Isolate,并执行指定的任务。
示例:创建 Isolate
import 'dart:isolate';void isolateTask(String message) {print('Isolate received: $message');
}void main() {print('Main isolate: Start');// 启动新的 IsolateIsolate.spawn(isolateTask, 'Hello from Main isolate');print('Main isolate: End');
}
输出:
Main isolate: Start
Main isolate: End
Isolate received: Hello from Main isolate
在这个例子中,我们使用 Isolate.spawn() 创建了一个新的 Isolate,运行 isolateTask() 函数,同时将消息传递给新的 Isolate。可以看到主线程不会等待 Isolate 的完成,而是继续执行后续代码。
Isolate 之间的消息传递
由于 Isolate 之间不能共享内存,因此它们只能通过 消息传递 进行通信。Dart 提供了 SendPort 和 ReceivePort 来在不同 Isolate 之间传递消息。
ReceivePort:接收消息的端口,类似于消息队列。SendPort:发送消息的端口,通过SendPort可以向另一个 Isolate 发送消息。
创建消息传递机制
首先需要在主 Isolate 创建一个 ReceivePort,并将 SendPort 传递给新的 Isolate。新的 Isolate 可以通过 SendPort 发送消息,主 Isolate 使用 ReceivePort 来接收消息。
示例:主 Isolate 和子 Isolate 间的消息传递
import 'dart:isolate';void isolateTask(SendPort sendPort) {// 向主 Isolate 发送消息sendPort.send('Message from Isolate');
}void main() async {// 创建用于接收消息的 ReceivePortReceivePort receivePort = ReceivePort();// 启动新的 Isolate,并传递 SendPortawait Isolate.spawn(isolateTask, receivePort.sendPort);// 监听来自 Isolate 的消息receivePort.listen((message) {print('Main isolate received: $message');});
}
输出:
Main isolate received: Message from Isolate
在这个例子中,我们创建了一个 ReceivePort,并将它的 SendPort 传递给新的 Isolate。子 Isolate 使用 sendPort.send() 发送消息,主 Isolate 则通过 receivePort.listen() 接收并处理消息。
Isolate 双向通信
除了子 Isolate 向主 Isolate 发送消息之外,主 Isolate 也可以向子 Isolate 发送消息。这需要双向的 SendPort 和 ReceivePort,实现双向通信。
实现双向通信
在双向通信中,主 Isolate 和子 Isolate 都有各自的 SendPort 和 ReceivePort,相互之间可以发送和接收消息。
示例:双向通信
import 'dart:isolate';// 子 Isolate 任务,接收消息并回复
void isolateTask(SendPort mainSendPort) {// 创建子 Isolate 的接收端口ReceivePort isolateReceivePort = ReceivePort();// 向主 Isolate 发送子 Isolate 的 SendPortmainSendPort.send(isolateReceivePort.sendPort);// 监听来自主 Isolate 的消息isolateReceivePort.listen((message) {print('Isolate received: $message');// 回复主 IsolatemainSendPort.send('Reply from Isolate');});
}void main() async {// 创建主 Isolate 的接收端口ReceivePort mainReceivePort = ReceivePort();// 启动子 Isolate,并传递主 Isolate 的 SendPortawait Isolate.spawn(isolateTask, mainReceivePort.sendPort);// 监听来自子 Isolate 的消息mainReceivePort.listen((message) {if (message is SendPort) {// 收到子 Isolate 的 SendPort,向其发送消息SendPort isolateSendPort = message;isolateSendPort.send('Hello from Main isolate');} else {print('Main isolate received: $message');}});
}
输出:
Isolate received: Hello from Main isolate
Main isolate received: Reply from Isolate
在这个示例中,主 Isolate 和子 Isolate 都有自己的 ReceivePort 和 SendPort。主 Isolate 将自己的 SendPort 传递给子 Isolate,子 Isolate 通过该 SendPort 发送消息回复主 Isolate。实现了双向的通信。
Isolate 的常见使用场景
耗时计算
在复杂的计算任务(如图像处理、大数据计算等)中使用 Isolate 可以避免阻塞 UI 线程。
示例:耗时任务
import 'dart:isolate';// 耗时任务
void computeTask(SendPort sendPort) {int sum = 0;for (int i = 0; i < 100000000; i++) {sum += i;}sendPort.send(sum);
}void main() async {ReceivePort receivePort = ReceivePort();// 启动 Isolate 执行耗时任务await Isolate.spawn(computeTask, receivePort.sendPort);// 获取计算结果receivePort.listen((result) {print('Sum: $result');});
}
在这个例子中,计算任务被移到子 Isolate 中执行,主线程不会被阻塞,从而保证了应用的流畅性。
网络请求并发处理
通过 Isolate 可以并行处理多个网络请求,提升网络任务的处理效率。
总结
Isolate 是 Dart 中一种重要的并行执行机制,适用于需要处理复杂计算或长时间执行任务的场景。与传统的多线程不同,Isolate 之间通过消息传递进行通信,避免了数据竞争和线程同步问题。在 Flutter 开发中,合理使用 Isolate 可以提高应用的性能和用户体验,确保长时间任务不会阻塞主线程。
掌握 Isolate 的使用,包括事件循环、消息传递和双向通信,可以帮助你构建高性能、响应迅速的应用。在实际开发中,Isolate 主要用于耗时操作、并发任务以及后台数据处理等场景。
相关文章:
【Flutter】Dart:Isolate
在 Dart 和 Flutter 中,所有的代码默认都运行在单一的线程(即主线程)上,这个线程也叫做 UI 线程。当进行耗时操作(如复杂计算或网络请求)时,如果不使用多线程处理,主线程会被阻塞&am…...
微信小程序 页面间传递数据
在小程序中,给页面传递参数通常有以下几种方法: 通过URL传递参数: 在小程序中,可以在页面的路径后面添加参数,然后在页面的 onLoad 函数中获取这些参数。 // 在app.json中配置页面路径 "pages": [{"pat…...
前端_005_Nodejs
文章目录 npm包管理器cjs和mjsYarn包管理器 1.Node.js 是js的一个运行环境,从nodejs诞生后js代码不局限于只在浏览器中执行,此外还能再nodejs里写服务端,用js可以前后端全栈开发 2.Node.js不跟浏览器一样默认含有document,window对象…...
SpringCache缓存介绍
1.为什么需要缓存 前台请求,后台先从缓存中取数据,取到直接返回结果,取不到时从数据库中取,数据库取到更新缓存,并返回结果,数据库也没取到,那直接返回空结果: 使用缓存是一个很…...
python实战(一)——iris鸢尾花数据集分类
一、任务背景 本文是python实战系列专栏的第一篇文章,我们将从分类开始由浅入深逐步学习如何使用python完成常规的机器学习/深度学习任务。iris数据集是经典的机器学习入门数据集,许多分类任务教程都会以这个数据集作为示例,它的数据量是150条…...
k8s-对命名空间资源配额
对k8s命名空间限制的方法有很多种,今天来演示一下很常用的一种 用的k8s对象就是ResourceQuota 一:创建命名空间 kubectl create ns test #namespace命名空间可以简写成ns 二: 对命名空间进行限制 创建resourcequota vim resourcequ…...
Failed to connect to github.com port 443
git push无法连接443端口 **问题1****方法一:取消代理设置**git命令 其他解决方案1. **设置 Git 使用 HTTP 而不是 HTTPS**2. **检查证书**3. **配置 Git 忽略 SSL 验证(不推荐)**4. **检查代理设置** 问题1 Failed to connect to github.com…...
【设计模式系列】简单工厂模式
一、什么是简单工厂模式 简单工厂模式(Simple Factory Pattern)是一种设计模式,其中包含一个工厂类,根据传入的参数不同,返回不同类的实例。这个工厂类封装了对象的创建逻辑,使得客户端代码可以从直接创建…...
给定一个正整数n随机生成n个字节即生成2n个十六进制数将其组成字符串返回secrets.token_hex(n)
【小白从小学Python、C、Java】 【考研初试复试毕业设计】 【Python基础AI数据分析】 给定一个正整数n 随机生成n个字节 即生成2n个十六进制数 将其组成字符串返回 secrets.token_hex(n) [太阳]选择题 根据题目代码,执行的结果错误的是? import secrets …...
[Gtk] 工程
MediaPlayer 可执行文件工程 结构 . ├── BUILD ├── ButtonHelper.cpp ├── ButtonHelper.h ├── CMakeLists.txt ├── DrawingAreaHelper.cpp ├── DrawingAreaHelper.h ├── layout.ui └── main.cpp CMakeLists.txt # 1) cmake basic cmake_minimum_r…...
基于Multisim的汽车尾灯控制电路设计与仿真
假设汽车尾部左右量测各有3个指示灯(用发光二极管模拟)1. 汽车正常运行时指示灯全灭;2.右转弯时,右侧3个指示灯按右循环顺序点亮;.3. 左转弯时,左侧3个指示灯按左循环顺序点亮;4.临时刹车时所有…...
Leetcode 3326. Minimum Division Operations to Make Array Non Decreasing
Leetcode 3326. Minimum Division Operations to Make Array Non Decreasing 1. 解题思路2. 代码实现 题目链接:3326. Minimum Division Operations to Make Array Non Decreasing 1. 解题思路 这一题的话就是要看出来题中给出的operation的本质事实上就是将任意…...
redo文件误删除后通过逻辑备份进行恢复
问题描述 开发同事让在一个服务器上查找下先前库的备份文件是否存在,如果存在进行下恢复。翻了服务器发现备份文件存在,多愁了一眼竟翻到了该备份文件于2024.6.17日恢复过的日志,赶紧和开发沟通说2024.6.17号已经恢复过了为啥还要恢复&#x…...
7805的输出电压如何调整?
7805稳压集成电路的输出电压通常是固定的,标称为5V。然而,在实际应用中,可以通过一些方法调整其输出电压,尽管这些调整方法可能会使电路变得更加复杂或需要额外的元件。以下是几种可能的调整方法: 1. 使用不同型号的稳…...
git命令使用一览【自用】
git常见操作: git initgit remote add master【分支名字】 gitgits.xxxxx【仓库中获取的ssh链接或者http协议的链接】检查远程仓库是否链接成功。 git remote -v出现以下画面就可以git pull,git push了...
MES系列-报表和分析
MES系列-报表和分析 MES系列文章目录 ISA-95制造业中企业和控制系统的集成的国际标准-(1) ISA-95制造业中企业和控制系统的集成的国际标准-(2) ISA-95制造业中企业和控制系统的集成的国际标准-(3) ISA-95制造业中企业和控制系统的集成的国际标准-(4) ISA-95制造业中企业和控制…...
如何在分布式环境中实现高可靠性分布式锁
目录 一、简单了解分布式锁 (一)分布式锁:应对分布式环境的同步挑战 (二)分布式锁的实现方式 (三)分布式锁的使用场景 (四)分布式锁需满足的特点 二、Redis 实现分…...
Vue基础(4)
自定义指令 除了默认设置的核心指令( v-model 和 v-show ), Vue 也允许注册自定义指令。在vue中使用directive来创建自定义指令 钩子函数 指令定义函数提供了几个钩子函数(可选): bind: 只调用一次,指令第一次绑定到元素时调用&…...
Redis高阶篇之Redis单线程与多线程
文章目录 0 前言1. 为什么Redis是单线程?1.1 Redis单线程1.2 为什么Redis3时代单线程快的原因1.3 使用单线程原因 2.为什么逐渐加入多线程呢?2.1 如何解决 3.redis6/7的多线程特性和IO多路复用入门3.1主线程和IO线程怎么协作完成请求处理的3.2 Unix网络编…...
【C++】STL——priority_queue优先级队列
目录 前言priority_queue的使用简单使用在OJ中的使用 priority_queue的模拟实现基本功能仿函数在这里插入图片描述 前言 上一节我们说了stack和queue这两种容器适配器,而priority_queue(优先级队列)同样也是属于容器适配器,它会优…...
树莓派超全系列教程文档--(62)使用rpicam-app通过网络流式传输视频
使用rpicam-app通过网络流式传输视频 使用 rpicam-app 通过网络流式传输视频UDPTCPRTSPlibavGStreamerRTPlibcamerasrc GStreamer 元素 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 使用 rpicam-app 通过网络流式传输视频 本节介绍来自 rpica…...
Auto-Coder使用GPT-4o完成:在用TabPFN这个模型构建一个预测未来3天涨跌的分类任务
通过akshare库,获取股票数据,并生成TabPFN这个模型 可以识别、处理的格式,写一个完整的预处理示例,并构建一个预测未来 3 天股价涨跌的分类任务 用TabPFN这个模型构建一个预测未来 3 天股价涨跌的分类任务,进行预测并输…...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...
vue3+vite项目中使用.env文件环境变量方法
vue3vite项目中使用.env文件环境变量方法 .env文件作用命名规则常用的配置项示例使用方法注意事项在vite.config.js文件中读取环境变量方法 .env文件作用 .env 文件用于定义环境变量,这些变量可以在项目中通过 import.meta.env 进行访问。Vite 会自动加载这些环境变…...
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数
高效线程安全的单例模式:Python 中的懒加载与自定义初始化参数 在软件开发中,单例模式(Singleton Pattern)是一种常见的设计模式,确保一个类仅有一个实例,并提供一个全局访问点。在多线程环境下,实现单例模式时需要注意线程安全问题,以防止多个线程同时创建实例,导致…...
LINUX 69 FTP 客服管理系统 man 5 /etc/vsftpd/vsftpd.conf
FTP 客服管理系统 实现kefu123登录,不允许匿名访问,kefu只能访问/data/kefu目录,不能查看其他目录 创建账号密码 useradd kefu echo 123|passwd -stdin kefu [rootcode caozx26420]# echo 123|passwd --stdin kefu 更改用户 kefu 的密码…...
如何更改默认 Crontab 编辑器 ?
在 Linux 领域中,crontab 是您可能经常遇到的一个术语。这个实用程序在类 unix 操作系统上可用,用于调度在预定义时间和间隔自动执行的任务。这对管理员和高级用户非常有益,允许他们自动执行各种系统任务。 编辑 Crontab 文件通常使用文本编…...
Golang——6、指针和结构体
指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...
适应性Java用于现代 API:REST、GraphQL 和事件驱动
在快速发展的软件开发领域,REST、GraphQL 和事件驱动架构等新的 API 标准对于构建可扩展、高效的系统至关重要。Java 在现代 API 方面以其在企业应用中的稳定性而闻名,不断适应这些现代范式的需求。随着不断发展的生态系统,Java 在现代 API 方…...
如何应对敏捷转型中的团队阻力
应对敏捷转型中的团队阻力需要明确沟通敏捷转型目的、提升团队参与感、提供充分的培训与支持、逐步推进敏捷实践、建立清晰的奖励和反馈机制。其中,明确沟通敏捷转型目的尤为关键,团队成员只有清晰理解转型背后的原因和利益,才能降低对变化的…...
