优化 Flutter 应用开发:探索 ViewModel 的威力
介绍
1.1 什么是 ViewModel?
ViewModel,顾名思义,就是视图的模型。在 Flutter 中,ViewModel 是一种用于管理视图状态和业务逻辑的重要概念。它承载了应用程序的核心功能,像是一个精心设计的控制中心,负责连接视图和数据模型,使得应用程序能够顺畅地运行。
想象一下,当你在手机上点击一个按钮时,背后发生了什么?按钮按下后,应用程序可能需要从服务器获取数据、更新用户界面、保存用户操作等等。这些复杂的任务需要一个地方来统一管理,而这个地方就是 ViewModel。它像是一个灵活的中介者,负责处理用户交互和数据流动,让应用程序保持良好的状态和流畅的运行。
1.2 ViewModel 的作用和重要性
ViewModel 的作用是多方面的,它扮演了多个角色:
-
管理视图状态:ViewModel 负责管理视图的状态,例如加载状态、错误状态、空数据状态等。它使得视图能够根据不同的状态显示不同的UI,提升用户体验。
-
处理业务逻辑:ViewModel 包含了应用程序的业务逻辑,例如数据处理、网络请求、数据持久化等。它将业务逻辑从视图中分离出来,使得代码更加清晰和易于维护。
-
与视图交互:ViewModel 通过与视图绑定来更新UI,例如通过 ChangeNotifier 实现状态监听,使得视图能够实时响应数据变化,保持界面的一致性和及时性。
因此,ViewModel 的重要性不言而喻。它是应用程序的核心架构之一,直接影响着应用程序的性能、可维护性和用户体验。
1.3 为什么在 Flutter 中需要 ViewModel?
在 Flutter 中,视图和业务逻辑通常紧密耦合在一起,这导致了一些问题:
- 代码混乱:视图代码和业务逻辑混杂在一起,使得代码难以理解和维护。
- 难以测试:由于视图和业务逻辑耦合在一起,使得单元测试变得困难,无法有效地测试业务逻辑。
- 重复代码:相似的业务逻辑可能被重复实现在多个视图中,导致了代码的冗余和低效。
因此,引入 ViewModel 架构可以解决这些问题:
- 分离关注点:ViewModel 将视图和业务逻辑分离,使得代码更加清晰和模块化,易于理解和维护。
- 提升可测试性:ViewModel 可以单独进行单元测试,保证业务逻辑的正确性,提升了代码的质量和稳定性。
- 提高重用性:ViewModel 可以被多个视图共享和重用,避免了重复实现相似的业务逻辑,提高了代码的复用率和可维护性。
因此,引入 ViewModel 架构是为了提升 Flutter 应用程序的可维护性、可测试性和用户体验,使得应用程序更加健壮和易于开发。
Flutter 中的状态管理概述
2.1 基本概念:State、StatefulWidget、ChangeNotifier 等
在 Flutter 中,状态管理是构建应用程序的关键部分。让我们先来了解一些基本概念:
-
State:状态是指应用程序中可能会变化的数据或信息,例如用户输入、网络响应、UI状态等。在 Flutter 中,状态通常被封装在 State 对象中,并由 StatefulWidget 来管理和更新。
-
StatefulWidget:StatefulWidget 是一个可变的组件,它可以根据不同的状态显示不同的UI。它包含一个对应的 State 对象,用于管理组件的状态和生命周期。
-
ChangeNotifier:ChangeNotifier 是 Flutter 提供的一个简单的状态管理类,它实现了一个发布-订阅模式,可以通知依赖它的组件进行更新。通常与 Provider 结合使用,用于构建可重用的状态管理解决方案。
2.2 常见的状态管理方案:setState、Provider、GetX、Riverpod 等
在 Flutter 中,有许多不同的状态管理方案可供选择,每种方案都有其自己的特点和适用场景:
-
setState:setState 是 Flutter 最基本的状态管理方案之一,它通过调用 setState 方法来更新组件的状态,并触发重新构建UI。虽然简单易用,但在大型应用程序中会导致代码冗余和复杂性增加。
-
Provider:Provider 是一个轻量级的状态管理库,它基于 InheritedWidget 和 ChangeNotifier 实现了依赖注入和状态通知。它具有简单、灵活、易于使用等特点,适用于中小型应用程序的状态管理。
-
GetX:GetX 是一个功能丰富的状态管理库,它提供了状态管理、路由管理、依赖注入等功能。它具有简洁、高性能、易于集成等特点,适用于需要快速开发的中大型应用程序。
-
Riverpod:Riverpod 是一个基于 Provider 的新一代状态管理库,它引入了更强大的依赖注入和异步操作支持。它具有类型安全、易于测试、灵活性高等特点,适用于复杂的大型应用程序。
每种状态管理方案都有其适用的场景和优缺点,开发者可以根据项目需求和个人偏好选择合适的方案。无论选择哪种方案,都需要考虑到项目的规模、复杂度和团队的技术水平,以及未来的可扩展性和维护性。
ViewModel 的设计原则
3.1 单一职责原则
在设计 ViewModel 时,我们要遵循单一职责原则,就像给一个人分配一个明确的任务一样。这意味着每个 ViewModel 应该专注于处理一个特定的功能或领域,而不是包揽所有的任务。
想象一下,如果一个人既要负责做饭又要负责洗衣服和打扫卫生,那么可能会出现混乱和效率低下的情况。同样地,一个 ViewModel 如果承担了太多的责任,就会变得臃肿和难以维护。
因此,我们应该将功能分解,每个 ViewModel 只负责一个明确的功能或领域,这样可以使得代码更加清晰、模块化和易于扩展。
3.2 数据驱动原则
ViewModel 的设计应该是数据驱动的,就像汽车的方向盘决定了汽车的前进方向一样。这意味着 ViewModel 应该根据数据的变化来驱动视图的更新,而不是直接操作UI元素。
想象一下,如果一个人不知道目的地在哪里,那么无论如何操作方向盘都不会有意义。同样地,一个 ViewModel 如果没有数据驱动,而是直接操作UI元素,那么就会导致代码混乱和耦合性增加。
因此,我们应该让数据成为驱动力,ViewModel 应该根据数据的变化来更新视图,保持代码的清晰和一致性。
3.3 可测试性原则
ViewModel 的设计应该具有良好的可测试性,就像一个产品经理提出的需求可以被快速验证一样。这意味着我们应该设计 ViewModel,使得它可以轻松地进行单元测试,验证其功能的正确性和稳定性。
想象一下,如果一个产品的功能无法被快速验证,那么可能会导致产品质量下降和用户体验差。同样地,一个无法进行单元测试的 ViewModel,可能会隐藏着许多潜在的问题和风险。
因此,我们应该设计 ViewModel,使其具有良好的单元测试覆盖率,保证其功能的正确性和稳定性,从而提高代码的质量和可维护性。
ViewModel 的实现方式
4.1 使用 Provider 实现 ViewModel
Provider 是 Flutter 中常用的状态管理库之一,它提供了简单而强大的状态管理功能,可以用来实现 ViewModel。使用 Provider 实现 ViewModel 的步骤如下:
- 创建 ViewModel 类:编写一个继承自 ChangeNotifier 的 ViewModel 类,定义视图状态和业务逻辑。
- 使用 ChangeNotifierProvider 提供 ViewModel:在顶层 Widget 中使用 ChangeNotifierProvider 提供 ViewModel 实例,使得整个应用程序都能访问到 ViewModel。
- 在视图中使用 Consumer 或 Provider.of 获取 ViewModel:在需要访问 ViewModel 的地方使用 Consumer 或 Provider.of 获取 ViewModel 实例,并根据需要更新视图。
4.2 使用 GetX 实现 ViewModel
GetX 是一个功能丰富的状态管理库,它提供了状态管理、路由管理、依赖注入等功能,并且使用起来非常简单和方便。使用 GetX 实现 ViewModel 的步骤如下:
- 创建 Controller 类:编写一个继承自 GetxController 的 Controller 类,定义视图状态和业务逻辑。
- 在视图中使用 GetBuilder 或 Obx 获取 Controller:在需要访问 Controller 的地方使用 GetBuilder 或 Obx 获取 Controller 实例,并根据需要更新视图。
4.3 使用 Riverpod 实现 ViewModel
Riverpod 是一个基于 Provider 的新一代状态管理库,它引入了更强大的依赖注入和异步操作支持。使用 Riverpod 实现 ViewModel 的步骤如下:
- 创建 ViewModel 类:编写一个普通的 Dart 类,定义视图状态和业务逻辑。
- 使用 Provider 或 ConsumerWidget 提供 ViewModel:在视图中使用 Provider 或 ConsumerWidget 提供 ViewModel 实例,并根据需要更新视图。
4.4 自定义实现 ViewModel
除了使用现有的状态管理库之外,我们还可以自定义实现 ViewModel。这种方式可以根据项目的需求和个人偏好来灵活定制,通常会更加灵活和可控。自定义实现 ViewModel 的步骤如下:
- 创建 ViewModel 类:编写一个普通的 Dart 类,定义视图状态和业务逻辑。
- 在视图中引入 ViewModel:在需要访问 ViewModel 的地方引入 ViewModel 类,并根据需要更新视图。
无论使用哪种方式实现 ViewModel,都需要考虑到项目的规模、复杂度和团队的技术水平,以及未来的可扩展性和维护性。选择合适的实现方式可以使得代码更加清晰、模块化和易于维护。
这种方式类型安全、易于测试,适用于复杂的大型应用程序的状态管理。
4.4 自定义实现 ViewModel
想象一下你是一个发明家,你可以根据自己的需求和想法设计出自己的工具。自定义实现 ViewModel 就像是你设计自己的工具一样,根据项目需求和个人偏好来实现。
自定义实现 ViewModel 可以根据具体的业务需求来设计,可以使用各种各样的技术和框架,例如使用 BLoC、Redux、MobX 等。这种方式灵活性高,可以根据项目的需求来选择合适的实现方式。
ViewModel 的最佳实践
5.1 如何设计和组织 ViewModel
设计和组织 ViewModel 是开发 Flutter 应用程序的关键步骤之一,它直接影响着代码的清晰度、可维护性和可扩展性。以下是一些最佳实践:
- 单一职责原则:每个 ViewModel 应该专注于处理一个特定的功能或领域,避免承担过多的责任,使得代码更加清晰和模块化。
- 分层结构:ViewModel 可以根据功能进行分层组织,例如将 UI 相关的逻辑和业务逻辑分开,使得代码更易于理解和维护。
- 抽象和接口:使用抽象类和接口来定义 ViewModel 的结构和行为,可以使得代码更加灵活和可扩展。
5.2 ViewModel 与业务逻辑的关系
ViewModel 与业务逻辑之间有着密切的关系,它们相辅相成,共同构建了应用程序的核心功能。以下是一些最佳实践:
- 业务逻辑的委托:ViewModel 应该将复杂的业务逻辑委托给其他类来处理,使得代码更加清晰和可测试。
- 数据处理和状态管理:ViewModel 负责管理视图状态和处理数据,例如加载状态、错误状态、数据处理等。
- 网络请求和数据持久化:ViewModel 可以包含网络请求和数据持久化的逻辑,但应该将其封装成可复用的方法,便于在多个视图中共享和重用。
5.3 ViewModel 与视图的交互方式
ViewModel 与视图之间的交互方式直接影响着应用程序的用户体验和性能。以下是一些最佳实践:
- 数据驱动视图:ViewModel 应该根据数据的变化来驱动视图的更新,而不是直接操作 UI 元素,这样可以保持代码的清晰和一致性。
- 状态监听和通知:ViewModel 可以使用状态监听或通知机制来通知视图更新,例如使用 ChangeNotifier 实现状态监听,保证视图能够及时响应数据变化。
- 事件处理和用户交互:ViewModel 可以处理用户交互和事件响应,例如点击事件、输入事件等,但应该将其封装成可复用的方法,便于在多个视图中共享和重用。
综上所述,设计和组织 ViewModel、与业务逻辑的关系以及与视图的交互方式是开发 Flutter 应用程序时需要考虑的重要因素,合理的设计和实践可以提高代码的质量、可维护性和用户体验。
ViewModel 的案例分析
6.1 构建一个简单的 Flutter 应用程序
让我们从头开始构建一个简单的 Flutter 应用程序,例如一个待办事项列表应用程序。该应用程序包含一个输入框用于添加新的待办事项,以及一个列表用于显示已添加的待办事项。
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'todo_list_view_model.dart';void main() {runApp(MyApp());
}class MyApp extends StatelessWidget {Widget build(BuildContext context) {return ChangeNotifierProvider(create: (context) => TodoListViewModel(),child: MaterialApp(title: 'Todo List App',theme: ThemeData(primarySwatch: Colors.blue,visualDensity: VisualDensity.adaptivePlatformDensity,),home: TodoListPage(),),);}
}class TodoListPage extends StatelessWidget {Widget build(BuildContext context) {return Scaffold(appBar: AppBar(title: Text('Todo List'),),body: TodoListView(),floatingActionButton: AddTodoButton(),);}
}class TodoListView extends StatelessWidget {Widget build(BuildContext context) {final viewModel = Provider.of<TodoListViewModel>(context);return ListView.builder(itemCount: viewModel.todoList.length,itemBuilder: (context, index) {final todo = viewModel.todoList[index];return ListTile(title: Text(todo),trailing: IconButton(icon: Icon(Icons.delete),onPressed: () => viewModel.removeTodo(index),),);},);}
}class AddTodoButton extends StatelessWidget {Widget build(BuildContext context) {return FloatingActionButton(onPressed: () {showDialog(context: context,builder: (BuildContext context) {final TextEditingController controller = TextEditingController();return AlertDialog(title: Text('Add Todo'),content: TextField(controller: controller,decoration: InputDecoration(hintText: 'Enter your todo'),),actions: <Widget>[TextButton(child: Text('Cancel'),onPressed: () => Navigator.of(context).pop(),),TextButton(child: Text('Add'),onPressed: () {final todo = controller.text;if (todo.isNotEmpty) {Provider.of<TodoListViewModel>(context, listen: false).addTodo(todo);}Navigator.of(context).pop();},),],);},);},tooltip: 'Add Todo',child: Icon(Icons.add),);}
}
6.2 使用 ViewModel 管理应用程序的状态和逻辑
我们可以使用 ViewModel 来管理应用程序的状态和逻辑。在这个案例中,我们创建一个名为 TodoListViewModel
的 ViewModel 类,它负责管理待办事项列表的状态和逻辑。
import 'package:flutter/material.dart';class TodoListViewModel extends ChangeNotifier {List<String> _todoList = [];List<String> get todoList => _todoList;void addTodo(String todo) {_todoList.add(todo);notifyListeners();}void removeTodo(int index) {_todoList.removeAt(index);notifyListeners();}
}
6.3 解决实际项目中的常见问题和挑战
在实际项目中,使用 ViewModel 可以解决许多常见的问题和挑战,例如:
- 数据管理:ViewModel 可以帮助我们管理应用程序的数据,包括加载数据、保存数据等。
- 状态管理:ViewModel 可以帮助我们管理应用程序的状态,例如加载状态、错误状态、空数据状态等。
- 业务逻辑:ViewModel 可以帮助我们管理应用程序的业务逻辑,例如处理用户交互、网络请求、数据处理等。
- 视图更新:ViewModel 可以帮助我们更新视图,保证视图能够及时响应数据变化,提升用户体验。
在这个案例中,我们使用 ViewModel 来管理待办事项列表的状态和逻辑,使得应用程序更加清晰、模块化和易于维护。同时,ViewModel 还可以帮助我们解决其他实际项目中的常见问题和挑战,提升开发效率和代码质量。
总结
使用 ViewModel 是一种有效地组织和管理应用程序代码的方法,它能够帮助我们优化状态管理、解耦视图和业务逻辑、提升开发效率和增强可测试性,是开发高质量 Flutter 应用程序的重要手段之一。
相关文章:
优化 Flutter 应用开发:探索 ViewModel 的威力
介绍 1.1 什么是 ViewModel? ViewModel,顾名思义,就是视图的模型。在 Flutter 中,ViewModel 是一种用于管理视图状态和业务逻辑的重要概念。它承载了应用程序的核心功能,像是一个精心设计的控制中心,负责…...

Android开发系列(四)Jetpack Compose之Button
在Jetpack Compose中,Button是一个常用的用户界面组件,用于执行某些操作或触发某些事件。Button控件是可触摸的,并且通常会显示一个文本或图标来表示其功能。 要在Jetpack Compose中创建一个Button,可以使用Button()函数…...

Java17 --- RabbitMQ之插件使用
目录 一、Federation插件 1.1、运行两个rabbitmq实例 1.2、启用插件 1.3、在下游端点添加上游端点 1.4、创建策略 1.6、测试 二、联邦队列 2.1、创建策略 2.2、创建交换机与队列 2.2.1、创建52000的队列与交换机 2.2.2、创建62000的队列 三、Shovel 3.1、启…...
6.18总结
省赛排位赛2: 省赛排名赛2 - Virtual Judge 思路: 设两个方程直接解出来就行 代码: #include<bits/stdc.h> using namespace std; int n, m; int main() {int n, m;int ans1, ans2;cin >> n >> m;ans1 n - (-3 sqr…...

【ARM Cache 及 MMU 系列文章 1.4 -- 如何判断 L3 Cache 是否实现?】
请阅读【ARM Cache 及 MMU/MPU 系列文章专栏导读】 及【嵌入式开发学习必备专栏】 文章目录 Cluster Configuration Register代码实现什么是Single-Threaded Core?什么是PE(Processor Execution units)?Single-Threaded Core与PE的关系对比多线程(Multithreading)Cluster…...
打印mybatis的sql日志
1、application.xml: logging.level.com.xxx.xxx.daodebug2、log4j2.xml: <Logger name"com.xxx.xxx.dao" level"debug" additivity"true" />...

QT day4(对话框 事件机制)
1:思维导图 2: #include "widget.h" #include "ui_widget.h"Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget) {ui->setupUi(this);ui->setupUi(this);//去除头部this->setWindowFlag(Qt::Frameles…...

序列化与反序列化漏洞实例
实验环境: 本次的序列化与反序列化漏洞为2021年强网杯上的一道比赛题目,我使用phpstudy集成环境将其测试环境搭建在了本地,如下。涉及的几个页面php为: index.php function.php myclass.php index.php : <?php // inde…...
6、while循环 - 习题解析
目录 解析部分:分支练习1244. 请问一个正整数能够整除几次2问题描述解题思路代码实现代码解析 1062. 求落地次数问题描述解题思路代码实现代码解析 1254. 求车速问题描述解题思路代码实现代码解析 1261. 韩信点兵问题描述解题思路代码实现代码解析 解析部分…...
ReentrantLock可重入锁
可重⼊锁,这个锁可以被线程多次重复进⼊进⾏获取操作。 ReentantLock继承接⼝Lock并实现了接⼝中定义的⽅法,除了能完成synchronized所能完成的所有⼯作 外,还提供了诸如可响应中断锁、可轮询锁请求、定时锁等避免多线程死锁的⽅法。 在并发量…...

如何秒杀系统架构设计
原文路径:https://learn.lianglianglee.com/%e4%b8%93%e6%a0%8f/%e5%a6%82%e4%bd%95%e8%ae%be%e8%ae%a1%e4%b8%80%e4%b8%aa%e7%a7%92%e6%9d%80%e7%b3%bb%e7%bb%9f/00%20%e5%bc%80%e7%af%87%e8%af%8d%20%e7%a7%92%e6%9d%80%e7%b3%bb%e7%bb%9f%e6%9e%b6%e6%9e%84%e8%ae%be%e8%ae%…...

深度神经网络——什么是降维?
引言 什么是降维? 降维是用于降低数据集维度的过程,采用许多特征并将它们表示为更少的特征。 例如,降维可用于将二十个特征的数据集减少到仅有几个特征。 降维通常用于无监督学习任务 降维是一个用于降低数据集维度的过程,采用许…...

SpringMVC—RequestMapping注解
一、RequestMapping注解 RequestMapping注解:是Spring MVC框架中的一个控制器映射注解,用于将请求映射到相应的处理方法上,具体来说,他可以将指定URL的请求绑定到一个特定的方法或类上,从而实现对请求的处理和响应。 …...

Java线程池基本概念
全局和局部线程池 全局线程池 在Spring框架中,全局线程池如ThreadPoolTaskExecutor通常是作为Spring Bean存在的,它们的生命周期由Spring容器管理。当Spring容器关闭时,这些线程池也会被适当地清理和关闭。因此,开发者通常不需要手…...
智能车联网安全发展形势、挑战
一、技术演进加速车联网安全环境复杂变化 当前,5G、大数据、大算力、大模型等技术正加速在车联网领域实现融合应用。车联网的网络通信能力、感知计算水平以及创新业务应用都在快速发展,与此同时,车联网的网络安全环境也在随之演进变化&#…...
AWS概述
AWS概述EMR Serverless Aamzon Web Services提供了一系列全球范围的云产品,包括计算、存储、数据库、分析、网络、移动、开发工具、管理工具、IoT、安全和企业应用:按需交付、及时可用、采用随用随付的定价模式。你可以畅享200多种服务,从数据…...

MySQL常见面试题自测
文章目录 MySQL基础架构一、说说 MySQL 的架构?二、一条 SQL语句在MySQL中的执行过程 MySQL存储引擎一、MySQL 提供了哪些存储引擎?二、MySQL 存储引擎架构了解吗?三、MyISAM 和 InnoDB 的区别? MySQL 事务一、何谓事务࿱…...

c语言回顾-函数递归
1.递归的介绍 1.1什么是递归 递归是指在一个函数的定义中调用自身的过程。简单来说,递归是一种通过重复调用自身来解决问题的方法。 递归包括两个关键要素:基本情况和递归情况。基本情况是指当问题达到某个特定条件时,不再需要递归调用&am…...

消息队列-RabbitMQ-延时队列实现
死信队列 DLX,全称为Dead-Letter-Exchange,死信交换机,死信邮箱。当消息在一个队列中变成死信之后,它能重新发送到另外一个交换器中,这个交换器就是DLX,绑定DLX的队列就称为死信队列。 导致死信的几种原因: ● 消息…...

【热门开源项目推荐】满足不同程序员的需求与关注点
目录 前言一、热门开源项目介绍二、使用开源热门项目的优势(一)经济方面(二)技术方面(三)社区支持及协作方面 三、程序员选择项目模型建议(一)关键步骤(二)示…...

基于FPGA的PID算法学习———实现PID比例控制算法
基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容:参考网站: PID算法控制 PID即:Proportional(比例)、Integral(积分&…...
树莓派超全系列教程文档--(62)使用rpicam-app通过网络流式传输视频
使用rpicam-app通过网络流式传输视频 使用 rpicam-app 通过网络流式传输视频UDPTCPRTSPlibavGStreamerRTPlibcamerasrc GStreamer 元素 文章来源: http://raspberry.dns8844.cn/documentation 原文网址 使用 rpicam-app 通过网络流式传输视频 本节介绍来自 rpica…...

【Oracle APEX开发小技巧12】
有如下需求: 有一个问题反馈页面,要实现在apex页面展示能直观看到反馈时间超过7天未处理的数据,方便管理员及时处理反馈。 我的方法:直接将逻辑写在SQL中,这样可以直接在页面展示 完整代码: SELECTSF.FE…...

转转集团旗下首家二手多品类循环仓店“超级转转”开业
6月9日,国内领先的循环经济企业转转集团旗下首家二手多品类循环仓店“超级转转”正式开业。 转转集团创始人兼CEO黄炜、转转循环时尚发起人朱珠、转转集团COO兼红布林CEO胡伟琨、王府井集团副总裁祝捷等出席了开业剪彩仪式。 据「TMT星球」了解,“超级…...

1.3 VSCode安装与环境配置
进入网址Visual Studio Code - Code Editing. Redefined下载.deb文件,然后打开终端,进入下载文件夹,键入命令 sudo dpkg -i code_1.100.3-1748872405_amd64.deb 在终端键入命令code即启动vscode 需要安装插件列表 1.Chinese简化 2.ros …...

QT: `long long` 类型转换为 `QString` 2025.6.5
在 Qt 中,将 long long 类型转换为 QString 可以通过以下两种常用方法实现: 方法 1:使用 QString::number() 直接调用 QString 的静态方法 number(),将数值转换为字符串: long long value 1234567890123456789LL; …...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
CMake控制VS2022项目文件分组
我们可以通过 CMake 控制源文件的组织结构,使它们在 VS 解决方案资源管理器中以“组”(Filter)的形式进行分类展示。 🎯 目标 通过 CMake 脚本将 .cpp、.h 等源文件分组显示在 Visual Studio 2022 的解决方案资源管理器中。 ✅ 支持的方法汇总(共4种) 方法描述是否推荐…...

处理vxe-table 表尾数据是单独一个接口,表格tableData数据更新后,需要点击两下,表尾才是正确的
修改bug思路: 分别把 tabledata 和 表尾相关数据 console.log() 发现 更新数据先后顺序不对 settimeout延迟查询表格接口 ——测试可行 升级↑:async await 等接口返回后再开始下一个接口查询 ________________________________________________________…...

Netty从入门到进阶(二)
二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架,用于…...