【Flutter】Dart:Isolate

news2024/10/23 13:54:16

在 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 中的异步操作(如 FutureStream)也都是通过事件循环来调度的。当一个 Isolate 接收到消息时,它会将消息放入事件队列,并在合适的时机进行处理。

如何创建 Isolate

可以通过 Isolate.spawn() 来创建新的 Isolate。该方法会启动一个新的 Isolate,并执行指定的任务。

示例:创建 Isolate

import 'dart:isolate';

void isolateTask(String message) {
  print('Isolate received: $message');
}

void main() {
  print('Main isolate: Start');

  // 启动新的 Isolate
  Isolate.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 提供了 SendPortReceivePort 来在不同 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 {
  // 创建用于接收消息的 ReceivePort
  ReceivePort receivePort = ReceivePort();

  // 启动新的 Isolate,并传递 SendPort
  await 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 发送消息。这需要双向的 SendPortReceivePort,实现双向通信。

实现双向通信

在双向通信中,主 Isolate 和子 Isolate 都有各自的 SendPortReceivePort,相互之间可以发送和接收消息。

示例:双向通信

import 'dart:isolate';

// 子 Isolate 任务,接收消息并回复
void isolateTask(SendPort mainSendPort) {
  // 创建子 Isolate 的接收端口
  ReceivePort isolateReceivePort = ReceivePort();

  // 向主 Isolate 发送子 Isolate 的 SendPort
  mainSendPort.send(isolateReceivePort.sendPort);

  // 监听来自主 Isolate 的消息
  isolateReceivePort.listen((message) {
    print('Isolate received: $message');

    // 回复主 Isolate
    mainSendPort.send('Reply from Isolate');
  });
}

void main() async {
  // 创建主 Isolate 的接收端口
  ReceivePort mainReceivePort = ReceivePort();

  // 启动子 Isolate,并传递主 Isolate 的 SendPort
  await 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 都有自己的 ReceivePortSendPort。主 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 主要用于耗时操作、并发任务以及后台数据处理等场景。

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

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

相关文章

构建后端为etcd的CoreDNS的容器集群(二)、下载最新的etcd容器镜像

在尝试获取etcd的容器的最新版本镜像时&#xff0c;使用latest作为tag取到的并非最新版本&#xff0c;本文尝试用实际最新版本的版本号进行pull&#xff0c;从而取到想的最新版etcd容器镜像。 一、用latest作为tag尝试下载最新etcd的镜像 1、下载镜像 [rootlocalhost opt]# …

多品牌摄像机视频平台EasyCVR海康大华宇视视频平台如何接入多样化设备

在实际的工程项目里&#xff0c;我们常常会面临这样的情况&#xff1a;项目管理者可能会决定使用多个品牌的视频监控摄像头&#xff0c;或者有需求将现有的、多种类型的监控系统进行整合。现在&#xff0c;让我们来探讨一下如何实现不同品牌摄像头的连接和使用。 1、GB/T281协议…

2024版最新148款CTF工具整理大全(附下载安装包)含基础环境、Web 安全、加密解密、密码爆破、文件、隐写、逆向、PWN

经常会有大学生粉丝朋友私信小强&#xff0c;想通过打CTF比赛镀金&#xff0c;作为进入一线互联网大厂的门票。 但是在CTF做题很多的时候都会用到工具&#xff0c;所以在全网苦寻CTF比赛工具安装包&#xff01; 关于我 有不少阅读过我文章的伙伴都知道&#xff0c;我曾就职于…

SSM框架实战小项目:打造高效用户管理系统 day3

前言 在前两篇博客中&#xff0c;后台已经搭建完毕&#xff0c;现在需要设计一下前端页面 webapp下的项目结构图 创建ftl文件夹&#xff0c;导入css和js 因为我们在后台的视图解析器中&#xff0c;设置了页面解析器&#xff0c;跳转路径为/ftl/*.ftl&#xff0c;所以需要ftl文件…

JAVA开源项目 网上订餐系统 计算机毕业设计

本文项目编号 T 018 &#xff0c;文末自助获取源码 \color{red}{T018&#xff0c;文末自助获取源码} T018&#xff0c;文末自助获取源码 目录 一、系统介绍二、演示录屏三、启动教程四、功能截图五、文案资料5.1 选题背景5.2 国内外研究现状5.3 可行性分析 六、核心代码6.1 新…

Pycharm通过ssh远程docker容器搭建开发环境

本地pycharm通过ssh方式远程连接服务器&#xff08;Ubuntu&#xff09;上的docker容器&#xff0c;搭建开发环境。实现功能&#xff1a;将环境依赖打包成docker镜像&#xff0c;并在远程服务器中启动容器&#xff08;启动时做好端口映射和文件夹挂载&#xff09;&#xff0c;通…

负载箱的作用?

负载箱&#xff0c;顾名思义&#xff0c;就是用来承载电力设备的箱子。在电力系统中&#xff0c;负载箱的作用非常重要&#xff0c;它不仅可以模拟实际的电力负载&#xff0c;还可以对电力设备进行测试和调试&#xff0c;确保其正常运行。下面详细介绍负载箱的作用。 1. 模拟实…

15分钟学Go 第7天:控制结构 - 条件语句

第7天&#xff1a;控制结构 - 条件语句 在Go语言中&#xff0c;控制结构是程序逻辑的重要组成部分。通过条件语句&#xff0c;我们可以根据不同的条件采取不同的行动。今天我们将详细探讨Go语言中的两种主要条件结构&#xff1a;if语句和switch语句。理解这些控制结构对于编写…

复写零--双指针

一&#xff1a;题目描述 题目链接&#xff1a;. - 力扣&#xff08;LeetCode&#xff09; 二&#xff1a;算法原理分析 三&#xff1a;代码编写 void duplicateZeros3(vector<int>& arr) {int dest -1, cur 0, n arr.size();//1.找到要复写的最后一个数字while …

qiankun 应用之间数据传递

qiankun 应用之间数据传递 全局共享 initGlobalState qiankun initGlobalState API 单击前往 qiankun 内部提供了 initGlobalState 方法用于注册 MicroAppStateActions 实例用于通信&#xff0c;该实例有三个方法&#xff0c;分别是onGlobalStateChange、setGlobalState、of…

微软宣布了新的“智能代理”功能

每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗&#xff1f;订阅我们的简报&#xff0c;深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同&#xff0c;从行业内部的深度分析和实用指南中受益。不要错过这个机会&#xff0c;成为AI领…

Elasticsearch:Redact(编辑) processor

Redact 处理器使用 Grok 规则引擎来隐藏输入文档中与给定 Grok 模式匹配的文本。该处理器可用于隐藏个人身份信息 (Personal Identifying Information - PII)&#xff0c;方法是将其配置为检测已知模式&#xff0c;例如电子邮件或 IP 地址。与 Grok 模式匹配的文本将被替换为可…

SpringBoot实现微信支付接口调用及回调函数(商户参数获取)

#1024程序员节 | 征文 # 一、具体业务流程 1. 用户下单 - 前端操作&#xff1a; - 用户在应用中选择商品、填写订单信息&#xff08;如地址、联系方式等&#xff09;&#xff0c;并点击“下单”按钮。 - 前端将订单信息&#xff08;商品ID、数量、价格等&#xff09;发送…

深度学习的一些数学基础

数学基础 万丈高楼平地起 怎么说呢&#xff0c;学的数二对于这些东西还是太陌生了&#xff0c;而且当时学的只会做题&#xff0c;不知道怎么使用/(ㄒoㄒ)/~~ 所以记下来一些不太清楚的前置知识点&#xff0c;主要来自《艾伯特深度学习》&#xff0c;书中内容很多&#xff0c…

动手学深度学习9.6. 编码器-解码器架构-笔记练习(PyTorch)

本节课程地址&#xff1a;61 编码器-解码器架构【动手学深度学习v2】_哔哩哔哩_bilibili 本节教材地址&#xff1a;9.6. 编码器-解码器架构 — 动手学深度学习 2.0.0 documentation (d2l.ai) 本节开源代码&#xff1a;...>d2l-zh>pytorch>chapter_multilayer-percep…

cnn_lstm_kan模型创新实现股票预测

获取更多完整项目代码数据集&#xff0c;点此加入免费社区群 &#xff1a; 首页-置顶必看 1. 项目简介 A002-cnn_lstm_kan模型创新实现股票预测项目旨在通过结合卷积神经网络&#xff08;CNN&#xff09;、长短期记忆网络&#xff08;LSTM&#xff09;以及知识注意网络&#…

Java项目实战II基于微信小程序UNIAPP+SSM+MySQL的电子点餐系统(开发文档+数据库+源码)

目录 一、前言 二、技术介绍 三、系统实现 四、文档参考 五、核心代码 六、源码获取 全栈码农以及毕业设计实战开发&#xff0c;CSDN平台Java领域新星创作者&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。获取源码联系方式请查看文末 一、前言 在当今快节…

华为配置 之 Console线路配置

目录 简介&#xff1a; 知识点&#xff1a; 配置Console线路密码 1.密码认证模式 2.AAA认证模式 知识点&#xff1a; 总结&#xff1a; 简介&#xff1a; 使用PC模拟器与路由器相连&#xff08;与交换机相连原理一样&#xff09;&#xff0c;在关机状态下&#xff0c;使用…

【CS常见问题】你用的是VS2019,最高支持.NET5.0,但是项目将.NET6.0设为目标无法运行,怎么办?

.NET版本问题 报错示例报错分析最简单的方法步骤 报错示例 严重性 代码 说明 项目 文件 行 禁止显示状态 错误 NETSDK1045 当前 .NET SDK 不支持将 .NET 6.0 设置为目标。请将 .NET 5.0 或更低版本设置为目标&#xff0c;或使用支持 .NET 6.0 的 .NET SDK 版本。 ABFview C:\x…

Request2:Post请求和Json

百度翻译拿到自己想看的数据&#xff0c;下图查看请求到数据的请求 preview提前看下 取出对应的RequestUrl &#xff0c;看出来要使用的话得用post请求 #!/usr/bin/env python # -*- coding:utf-8 -*- import requests import json if __name__ "__main__":#1.指定…