代理模式 - Flutter中的智能替身,掌控对象访问的每一道关卡!

痛点场景:直接加载高清大图

假设你的应用需要显示用户相册:

NetworkImage('https://example.com/high-res-photo.jpg')

面临的问题:

  • 📶 网络差时长时间白屏
  • 💾 重复下载相同图片浪费流量
  • 🔒 敏感图片无权限验证
  • 📱 内存占用过高导致崩溃

代理模式解决方案

核心思想: 为其他对象提供一种代理以控制对这个对象的访问。

三种常见代理类型:

  1. 虚拟代理: 延迟加载大资源(如占位图→高清图)
  2. 保护代理: 控制访问权限
  3. 缓存代理: 存储请求结果避免重复计算/下载

Flutter图片加载代理实现

1. 定义图片加载接口
abstract class ImageLoader {Future<ImageProvider> load(String url);
}
2. 实现真实图片加载器
class RealImageLoader implements ImageLoader {Future<ImageProvider> load(String url) async {print('真实加载器:从网络加载高清图片');return NetworkImage(url);}
}
3. 创建智能代理
class SmartImageLoader implements ImageLoader {final RealImageLoader _realLoader = RealImageLoader();final Map<String, ImageProvider> _cache = {};Future<ImageProvider> load(String url) async {// 1. 检查缓存if (_cache.containsKey(url)) {print('代理:从缓存返回图片');return _cache[url]!;}// 2. 显示占位图print('代理:先返回低分辨率占位图');final placeholder = ResizeImage(NetworkImage('$url?thumb=true'),width: 100,height: 100,);// 3. 后台加载高清图_realLoader.load(url).then((highResImage) {print('代理:高清图加载完成,更新缓存');_cache[url] = highResImage;// 通知UI更新(可通过ChangeNotifier或Stream)});return placeholder;}
}
4. 在Flutter中使用
class PhotoViewer extends StatelessWidget {final ImageLoader loader = SmartImageLoader(); // 使用代理Widget build(BuildContext context) {return FutureBuilder<ImageProvider>(future: loader.load('https://example.com/photo1.jpg'),builder: (ctx, snapshot) {if (snapshot.hasData) {return Image(image: snapshot.data!);}return CircularProgressIndicator();},);}
}

Flutter中的实际应用场景

场景1:权限控制代理
class AuthImageLoader implements ImageLoader {final UserService _userService;final RealImageLoader _realLoader;Future<ImageProvider> load(String url) async {if (!await _userService.checkPhotoPermission(url)) {return AssetImage('assets/locked.png');}return _realLoader.load(url);}
}// 使用
ImageLoader loader = AuthImageLoader(userService, realLoader);
场景2:API请求缓存代理
class CachedApiClient implements ApiClient {final ApiClient _realClient;final Map<String, dynamic> _cache = {};Future<dynamic> fetchData(String endpoint) async {if (_cache.containsKey(endpoint)) {return _cache[endpoint];}final data = await _realClient.fetchData(endpoint);_cache[endpoint] = data;return data;}
}// 使用
final apiClient = CachedApiClient(RealApiClient());
final product = await apiClient.fetchData('/products/123');
场景3:敏感操作延迟代理
class ExpenseReportGeneratorProxy implements ReportGenerator {RealReportGenerator? _realGenerator;Future<Report> generate() async {if (!await _checkPermission()) throw UnauthorizedException();_realGenerator ??= RealReportGenerator(); // 按需创建return _realGenerator!.generate();}
}

代理模式与Flutter状态管理结合

将图片代理与Provider结合:

class ImageLoaderProvider extends ChangeNotifier {final ImageLoader _loader;ImageProvider? _currentImage;ImageLoaderProvider(this._loader);Future<void> loadImage(String url) async {_currentImage = await _loader.load(url);notifyListeners();}
}// 使用
context.read<ImageLoaderProvider>().loadImage('photo.jpg');Consumer<ImageLoaderProvider>(builder: (context, provider, child) {return provider.currentImage != null? Image(image: provider.currentImage!): Placeholder();}
)

代理模式最佳实践

  1. 何时使用代理模式:

    • 需要延迟加载大资源时(虚拟代理)
    • 需要控制原始对象访问权限时(保护代理)
    • 需要缓存请求结果时(缓存代理)
    • 需要记录日志或监控对象访问时
  2. Flutter特化技巧:

    // 组合多种代理
    ImageLoader loader = LoggingImageLoader(CacheImageLoader(AuthImageLoader(RealImageLoader()))
    );// 使用Futures组合
    Future<ImageProvider> load(String url) async {final lowRes = await _loadLowRes(url);final highRes = _loadHighRes(url); // 不awaitreturn ProxyImage(lowRes, highRes);
    }
  3. 性能优化:

    // 预加载代理
    class PrefetchImageLoader implements ImageLoader {final List<String> _prefetchUrls;void prefetchAll() {for (final url in _prefetchUrls) {_realLoader.load(url); // 不await,后台加载}}
    }
  4. 测试策略:

    test('缓存代理测试', () async {final mockLoader = MockImageLoader();final proxy = CachedImageLoader(mockLoader);when(mockLoader.load('test.jpg')).thenAnswer((_) async => FakeImage());// 第一次调用真实对象await proxy.load('test.jpg');// 第二次应从缓存返回await proxy.load('test.jpg');verify(mockLoader.load('test.jpg')).called(1); // 仅调用一次
    });

代理模式 vs 装饰器模式

特性代理模式装饰器模式
目的控制访问增强功能
关系代理与真实对象关系固定装饰器可递归嵌套
创建时机通常提前知道真实对象运行时动态组合
典型应用懒加载、权限控制、缓存流式处理、动态扩展

代理模式的高级变体

1. 远程代理(跨进程通信)
// 本地代理
class LocalImageProxy implements ImageLoader {Future<ImageProvider> load(String url) async {// 通过平台通道调用原生代码final result = await MethodChannel('image_loader').invokeMethod('loadImage', {'url': url});return MemoryImage(base64Decode(result));}
}
2. 智能引用代理
class ImageRefProxy implements ImageLoader {int _refCount = 0;ImageProvider? _cachedImage;Future<ImageProvider> load(String url) async {_refCount++;if (_cachedImage == null) {_cachedImage = await _realLoader.load(url);}return _cachedImage!;}void release() {_refCount--;if (_refCount == 0) {_cachedImage?.evict(); // 释放内存_cachedImage = null;}}
}
3. 日志监控代理
class AnalyticsImageLoader implements ImageLoader {final AnalyticsService _analytics;final ImageLoader _realLoader;Future<ImageProvider> load(String url) async {_analytics.logEvent('image_load_start', {'url': url});final stopwatch = Stopwatch()..start();try {final image = await _realLoader.load(url);_analytics.logEvent('image_load_success', {'url': url,'duration': stopwatch.elapsedMilliseconds,});return image;} catch (e) {_analytics.logError('image_load_failed', {'url': url});rethrow;}}
}

总结:代理模式是你的访问管家

  • 核心价值: 在访问真实对象前后插入额外逻辑
  • Flutter优势:
    • 实现图片懒加载和缓存
    • 控制敏感资源访问
    • 减少网络请求和计算开销
    • 添加监控和日志记录
  • 适用场景: 图片加载、API调用、权限管理、性能优化

🕶️ 设计启示: 当你需要在访问对象时加入"中间层"控制时,代理模式就是你的"隐形眼镜",既能看清需求,又能保护眼睛!

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

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

相关文章

Python集合的创建

一、前言 在 Python 编程中&#xff0c;集合&#xff08;set&#xff09;是一种非常实用的数据结构&#xff0c;它能够存储一组无序且不重复的元素。集合广泛应用于数据去重、交并差运算等场景。 本文将重点讲解 Python 中集合的创建方式&#xff0c;包括使用大括号 {}、set(…

作物生长模型Oryza V3实战15:AutoCalibration程序详解

ORYZA 模型中的 AutoCalibration (v2.1).exe 是用于 ORYZA 模型参数自动校准的可执行程序,在优化 ORYZA 模型参数、提高模型模拟准确性方面具有重要作用。程序能够通过特定算法,在给定的参数取值范围内,自动搜索出一组最优的参数组合,使得模型模拟结果与实际观测数据(如作…

算法-每日一题(DAY12)最长和谐子序列

1.题目链接&#xff1a; 594. 最长和谐子序列 - 力扣&#xff08;LeetCode&#xff09; 2.题目描述&#xff1a; 和谐数组是指一个数组里元素的最大值和最小值之间的差别 正好是 1 。 给你一个整数数组 nums &#xff0c;请你在所有可能的 子序列 中找到最长的和谐子序列的…

阿里云-云效自动部署spring boot项目

1.使用云效通过docker自动部署spring boot项目 1.1 spring boot项目配置 # 阿里云的jdk17镜像 FROM registry.cn-zhangjiakou.aliyuncs.com/publicci/openjdk:17-jdk-alpineENV APP_HOME /home/admin/app/# 将target/arms-application.jar 复制到容器中 /home/admin/app/app.…

SQL篇 添加约束、删除约束

SQL篇 添加约束、删除约束 1、相关链接2、约束的增删找查2.1 查看约束&#xff08;主键、外键、唯一性、检查约束&#xff09;2.2 查看默认约束2.3 修改约束&#xff08;添加/编辑/修改&#xff09;2.3.1 添加主键约束2.3.2 添加外键约束2.3.3 添加唯一性约束2.3.4 添加检查约束…

Python PyTorch 深度学习库 包 timm

文章目录 &#x1f4e6; 主要特点&#x1f680; 安装方式&#x1f9ea; 使用示例示例1&#xff1a;加载一个预训练模型进行图像分类示例2&#xff1a;获取模型结构信息 &#x1f310; 官方资源&#x1f50d; 常见用途✅ 优势总结 Timm 是一个非常流行且功能强大的 Python 深度学…

tree 命令集成到 Git Bash:可视化目录结构的指南

目录 1. 下载与准备 tree 工具   2. 集成 tree 到 Git Bash 环境   3. tree 命令基础用法详解   4. 使用示例 在软件开发和文件管理中&#xff0c;清晰的目录结构可视化是提高效率的重要手段。tree命令作为 UNIX/Linux 系统的标准工具&#xff0c;能以树形结构递归展…

如何搭建基于RK3588的边缘服务器集群?支持12个RK3588云手机

以下是基于RK3588搭建边缘服务器集群的完整实施方案&#xff0c;涵盖硬件选型、集群架构、软件部署及优化要点&#xff1a; &#x1f5a5;️ ‌一、硬件集群架构设计‌ ‌节点基础配置‌ ‌核心单元‌&#xff1a;单节点采用RK3588核心板&#xff08;4A762.4GHz 4A551.8GHz&am…

飞算 JavaAI:我的编程强力助推引擎

文章目录 引言&#xff1a;当Java开发遇上AI助手初识飞算JavaAI&#xff1a;专为Java而生的智能伴侣安装与配置&#xff1a;轻松上手的开始核心功能体验&#xff1a;从需求到代码的全流程革命1. 智能需求分析与拆解2. 智能接口设计3. 表结构智能生成4. 处理逻辑自动梳理5. 高质…

飞算JavaAI—AI编程助手 | 编程领域的‘高科技指南针’,精准导航开发!

目录 一、引言 1.1 什么是飞算JavaAI&#xff1f; 1.2 告别"996的孤独感"&#xff1a;AI成为你的编码搭子 1.3 成就感加速器&#xff1a;从"能运行"到"优雅实现" 1.4 极简下载体验&#xff1a;3步开启"开挂"模式 二、深入体验飞…

NPM组件 betsson 等窃取主机敏感信息

【高危】NPM组件 betsson 等窃取主机敏感信息 漏洞描述 当用户安装受影响版本的 betsson 组件包时会窃取用户的主机名、用户名、工作目录、IP地址等信息并发送到攻击者可控的服务器地址。 MPS编号MPS-2nrw-lifd处置建议强烈建议修复发现时间2025-06-30投毒仓库npm投毒类型主…

Apipost 与 Apifox:API 开发管理中的 AI 能力对比

在当今竞争激烈的 API 开发与测试领域&#xff0c;效率与质量是衡量工具优劣的关键指标。Apipost 凭借其强大的 AI 功能&#xff0c;为开发者和测试人员带来了前所未有的便利&#xff0c;而 Apifox 作为该领域的重要参与者&#xff0c;二者在实际应用中究竟有何差异&#xff1f…

Electron 菜单栏深度定制指南:从基础到高级实践

在现代桌面应用开发中&#xff0c;菜单栏作为用户界面的重要组成部分&#xff0c;不仅提供了应用功能的快速访问途径&#xff0c;还直接影响着用户的操作体验。Electron 作为跨平台桌面应用开发框架&#xff0c;为开发者提供了强大而灵活的菜单系统定制能力。本文将全面介绍 El…

QML通过XMLHttpRequest实现HTTP通信

转自个人博客 由于 QML 的 JavaScript 兼容性&#xff0c;我们可以直接使用 JavaScript 的 XMLHttpRequest 对象进行 HTTP 请求。QML 的 XMLHttpRequest 实现与标准浏览器的实现非常相似&#xff0c;但有一些限制和特殊行为需要注意。 而QML实现TCP等其他通信一般就需要借助Qt与…

Spring Boot 内置反向代理(Undertow Proxy)高可用配置

引言 在微服务架构中&#xff0c;反向代理是一个不可或缺的组件&#xff0c;它负责请求转发、负载均衡、安全过滤等关键功能。 通常我们会选择 Nginx、HAProxy 等专业反向代理组件&#xff0c;但在某些场景下&#xff0c;使用 Spring Boot 内置的反向代理功能可以简化架构&am…

ClickHouse 部署

Docker 部署 1、拉取镜像 docker pull clickhouse/clickhouse-server:latest单机版本部署 编写docker-compose.yml version: 3services:clickhouse-server:image: clickhouse/clickhouse-server:22.12container_name: clickhouse-serverports:- "8123:8123"ulimit…

Fiddler中文版抓包工具如何帮助前端开发者高效调试

前端开发早已不再是“写好页面就完事”的工作。随着业务复杂度提升&#xff0c;前端开发者需要直面接口联调、性能优化、跨域排查、HTTPS调试等一系列和网络请求紧密相关的任务。抓包工具成为这些环节中不可替代的得力助手&#xff0c;而 Fiddler抓包工具 因其全面的功能和灵活…

WTL 之trunk技术学习

相比于MFC的消息机制&#xff0c;WTL/ATL的实现更加优雅。后者将win32 API与面向对象技术完美地结合起来&#xff0c;去掉了庞杂的MFC依赖&#xff0c;生成的软件体积更小&#xff0c;运行速度更快。在其中&#xff0c;如何将窗口函数转变为对窗口对象成员函数的调用&#xff0…

Linux——11.软件安装与包管理

Linux 与 Windows 系统在软件安装方式上的差异 Linux: Linux 通过 包管理系统(如 Debian 的 apt、Red Hat 的 yum/dnf)将软件打包为二进制安装包(如 .deb、.rpm),每个包包含程序文件、依赖关系和元数据。包管理系统负责统一管理软件的安装、更新、卸载,并自动处理依赖关…

无人机用shell远程登录机载电脑,每次需要环境配置原因

原因&#xff1a; 终端分为“登录 shell”和“非登录 shell”&#xff1a; - 登录 shell&#xff08;如开机登录、远程 SSH 连接&#xff09;会加载 .profile 或 .bash_profile 。 - 非登录 shell&#xff08;如打开新终端窗口&#xff09;会加载 .bashrc 。 - 如果环境变量…