当前位置: 首页 > news >正文

大师学SwiftUI第16章 - UIKit框架集成

其它相关内容请见​​虚拟现实(VR)/增强现实(AR)&visionOS开发学习笔记​​

SwiftUI是一套新框架,因此并没有包含我们构建专业应用所需的所有工具。这意味着我们会需要求助于UIKit(移动设备)和AppKit(Mac电脑)等原生框架所提供的工具。

我们已介绍过UIKit。它是一套SwiftUI在后台实现用于创建大部分视图和控件的框架。有些UIKit类用于运行应用(​​UIApplication​​​)、加载图片(​​UIImage​​​)、管理设备(​​UIDevice​​​)及窗口(​​UIWindow​​​),而另一些定义用于设置应用和场景(​​UiApplicationDelegate​​​和​​UIWindowSceneDelegate​​​)的代理。当然,该框架提供了我们创建界面所需的所有工具,包括两个创建和管理视图的基本类,称为​​UIView​​​和​​UIViewController​​。

最后面这两个类,​​UIView​​​和​​UIViewController​​,是在我们需要对SwiftUI界面添加UIKit特性时需要实现的类。UIView的子类用于在屏幕上显示信息,比如标签和图像,以及创建控件,比如按钮、滑块和开关。而​​UIViewController​​类的子类用于展示视图及包含处理它们的值和与用户交互的必要功能。为了将这些工具集成到SwiftUI界面中,SwiftUI框架定义了两个协议:​​UIViewRepresentable​​​和​​UIViewControllerRepresentable​​。

Representable视图

​UIViewRepresentable​​​协议定义了一结构体,作为​​UIView​​类或其子类所创建对象的一层封装(wrapper)。实现这一协议的结构体可表示SwiftUI界面中的一个UIKit视图。创建及管理UIKit视图,结构体必须要实现如下的方法。

  • ​makeUIView(context: Context)​​:该方法创建一个UIKit视图并返回。​​context​​​参数是对提供视图状态信息的​​UIViewRepresentableContext​​类型结构体的引用。
  • ​updateUIView(UIViewType, context: Context)​​:该方法通过一个绑定属性使用SwiftUI界面所提供的信息更新UIKit视图。第一个参数是对UIKit视图的引用,​​context​​​参数是对提供视图状态信息的​​UIViewRepresentableContext​​类型结构体的引用。
  • ​dismantleUIView(UIViewType, coordinator: Coordinator)​​:该类型方法为释放视图做准备工作。第一个参数是UIKit视图的指针,​​coordinator​​参数是将值发回SwiftUI界面的对象。
  • ​makeCoordinator()​​:该方法创建的对将UIKit视图的信息传回给SwiftUI界面。

要在SwiftUI接口中包含一个​​UIView​​​对象(或其子类所创建对象),我们需要定义一个实现​​UIViewRepresentable​​​协议的结构体,并实现上述的方法。​​makeUIView()​​​和​​updateUIView()​​​方法是必选的。在​​makeUIView()​​方法中,我们必须创建并返回一个UIKit视图实例,​​updateUIView()​​法用于使用SwiftUI界面中所获取的值来更新视图。

下例中创建了一个蓝色背景的UIKit视图,并在SwiftUI视图中包含它。只需要用​​makeUIView()​​方法创建UIKit视图,但同时也要实现​​updateUIView()​​方法,因为协议有相应要求。

示例16-1:准备一个供SwiftUI使用的UIKit视图

struct MyCustomView: UIViewRepresentable {func makeUIView(context: Context) -> some UIView {let view = UIView()view.backgroundColor = UIColor(.blue)return view}func updateUIView(_ uiView: UIViewType, context: Context) {}
}

在每次创建新的​​MyCustomView​​​结构体实例时调用​​makeUIView()​​​方法。在该方法中,我们创建一个​​UIView​​​视图,将背景色设置为蓝色并返回。因此,每次我们创建​​MyCustomView​​​结构体实例的时候,会创建一个​​UIView​​并添加到SwiftUI实例之中,如下例所示。

示例16-2:在SwiftUI视图中显示UIKit视图

struct ContentView: View {var body: some View {VStack {MyCustomView().frame(width: 200, height: 150).padding()Spacer()}}
}

​representable​​视图默认为弹性大小,但可以通过SwiftUI修饰符进行修改。本例中,我们使用frame()修饰符赋了固定的宽和高。

图16-1:SwiftUI界面中的UIKit视图

图16-1:SwiftUI界面中的UIKit视图

​UIView​​​对象创建一个空视图,但我们也可以实现接收其它用户输入的视图,如文本框、开关等。我们使用​​updateUIView()​​方法将SwiftUI视图中的值传递给UIKit视图。但如果想将UIKit视图中的值传递给SwiftUI界面,则需要实现​​makeCoordinator()​​​方法。在这个方法中,我们必须创建一个​​coordinator​​​对象实例并返回。​​coordinator​​是一个可将UIKit视图中的信息回发给Swift界面的对象,通常通过修改绑定属性实现。如何处理这些值取决于我们操作的UIkit视图的类型。例如,​​UITexView​​视图创建一个用户可输入多行文本的输入框,和SwiftUI中的​​TextEditor​​​视图一样。这个视图通过调用代理方法上报改变。因此,要获取用户在​​UITexView​​中插入的文本并在SwiftUI中进行处理,我们需要创建一个实现​​UITextViewDelegate​​​协议的​​coordinator​​​类,并实现其方法。下例演示了如何创建一个​​UIViewRepresentable​​结构体来操作这个类。

示例16-3:对SwiftUI视图发送和接收值

import SwiftUIstruct TextView: UIViewRepresentable {@Binding var input: Stringfunc makeUIView(context: Context) -> UITextView {let view = UITextView()view.backgroundColor = UIColor.yellowview.font = UIFont.systemFont(ofSize: 17)view.delegate = context.coordinatorreturn view}func updateUIView(_ uiView: UITextView, context: Context) {uiView.text = input}func makeCoordinator() -> CoordinatorTextView {return CoordinatorTextView(input: $input)}
}class CoordinatorTextView: NSObject, UITextViewDelegate {@Binding var inputCoordinator: Stringinit(input: Binding<String>) {self._inputCoordinator = input}func textViewDidChange(_ textView: UITextView) {inputCoordinator = textView.text}
}

本示例创建了一个名为​​TexView​​​的​​UIViewRepresentable​​​结构体。结构体中有一个名为​​input​​​的​​@Binding​​属性,接收值并传递给SwiftUI视图。在​​TexView​​​结构体定义的下面,我们定义了一类名为​​CoordinatorTextView​​​的类。这是我们视图的​​coordinator​​,任务是将值发回给SwiftUI视图。为此,我们通过一个关联​​TexView​​​结构体中定义的input属性的​​@Binding​​​属性对其初始化,然后实现一个代理方法,在用户插入或删了字符时执行。该方法获取输入框中的当前文本,因此我们将其赋值给那个​​@Binding​​属性,发回到SwiftUI视图。​​UIViewRepresentable​​​结构体初始化时。​​CoordinatorTextView​​​对象通过​​makeCoordinator()​​方法创建,因而该结构体在一开始就可接收和发送值了。

SwiftUI界面中的视图实现和之前相同。只需对存储添加一个​​@State​​​属性,将值传递给​​input​​属性。

示例16-4:对UIKit视图接收和发送值

struct ContentView: View {@State private var inputText: String = "Initial text"var body: some View {VStack {HStack {Text(inputText)Spacer()Button("Clear") {inputText = ""}}TextView(input: $inputText)}.padding()}
}

以上视图定义了一个名为​​inputText​​​的​​@State​​​属性、一个显示其值的​​Text​​​视图、一个用空字符串替换当前值的按钮以及自定义​​TextInput​​​视图的一个实例。该视图获取到​​inputText​​​属性的指针,与​​representable​​视图中的输入属性相关联,这样便可以来回传递值 。

用户在编辑器中输入或删除字符时,​​representable​​​视图调用​​coordinator​​​中的​​textViewDidChange()​​​方法,该方法将文本视图中的当前值赋给​​inputCoordinator​​​属性,也就赋给了​​input​​​属性。这意味着值在​​@State​​​属性中可用 ,并且​​Text​​​视图可将其显示到屏幕上。而在用户点击​​Clear​​​按钮时,我们将空字符串赋值给​​@State​​​属性,系统执行​​representable​​​视图中的​​updateUIView()​​​方法,关联​​@State​​​属性的​​@Binding​​属性的值被赋给了视图,文本视图被清除。

图16-2:SwiftUI界面中的UITextView视图

图16-2:SwiftUI界面中的​​UITexView​​视图

✍️跟我一起做:创建一个多平台项目。根据示例16-3中的代码创建一个名为​​TextView.swift​​​的文件。根据示例16-4中的代码更新​​ContextView​​​视图。插入一段文本。应该会看到顶部的文本发生了变化。点击​​Clear​​按钮。会看到文本视图及顶部的视图中的文本被删除。

注意:​​UIViewRepresentable​​结构体可表示任意UIKit视图。实现会取决于想使用的视图。我们会在后续的章节中看到更多的示例。要学习UIKitUiKit视图,可参阅UIKit for MasterMinds等书。

Representable视图控制器

​UIViewControllerRepresentable​​​协议定义一个结构体,作为对​​UIViewController​​类及子类所创建对象的封装。该类展示一个可包含其它视图的视图,为窗口或整个屏幕定义界面,类似SwiftUI文件中所定义的视图。实现​​UIViewControllerRepresentable​​协议的结构体可在一个SwiftUI界面中展示这些视图控制器。创建及管理这个视图控制器,结构体必须实现如下方法。

  • ​makeUIViewController(context: Context)​​:该方法创建一个UIKit视图控制器并返回。​​context​​​参数是​​UIViewControllerRepresentableContext​​类型结构体的一个指针,提供视图控制器状态的相关信息。
  • ​updateUIViewController(UIViewControllerType, context: Context)​​:该方法使用SwiftUI界面所提供的信息更新UIKit视图控制器。第一个参数是UIKit视图控制器的指针,​​context​​​参数是​​UIViewControllerRepresentableContext​​类型结构体的一个指针,提供视图控制器状态的相关信息。
  • ​dismantleUiViewController(UIViewControllerType, coordinator: Coordinator)​​​:该类型方法为释放视图控制器做准备。第一个参数是UIKit视图控制器的指针,​​coordinator​​参数是是将值发回SwiftUI界面的对象。
  • ​makeCoordinator()​​:该方法创建将UIKit视图控制器信息传回SwiftUI界面的对象。

UIKit视图控制器由​​UIViewController​​的子类创建。文件通过File菜单创建 ,但选取UIKit子类的选项称为Cocoa Touch Class。选择该选项后,Xcode会显示一个可输入文件名称的窗口,并可选择所创建子类相应的父类。本例中我们创建了一个继承​​UIViewController​​​类的​​DetailViewController​​类。

示例16-5:创建一个UIKit视图控制器

import UIKitclass DetailViewController: UIViewController {override func viewDidLoad() {super.viewDidLoad()let label = UILabel()label.frame = CGRect(x: 20, y: 16, width: 250, height: 30)label.font = UIFont.systemFont(ofSize: 30)label.text = "Hello World!"view.addSubview(label)}
}

在初始化视图控制器类时,它创建一个表示界面的视图,将其赋值给​​view​​​属性,并调用​​viewDidLoad()​​​方法来告诉我们的代码视图已就绪。在这个方法中,我们可以执行所有必要的任务来初始化视图。在本例中,我们创建一个​​UILabel​​对象来在屏幕上显示文本。该对象类似SwiftUI的​​Text​​​视图,但需要做一些配置。在我们的例子中,通过将​​CGRect​​​值赋给​​frame​​​属性来为其设置位置和大小,字体大小定义为30像素,赋值给要显示的文本,通过​​addSubview()​​方法将其添加给视图控制器的视图。

现在有了视图控制器,我们需要创建一个​​representable​​视图将其转为SwiftUI视图,如下例所示:

示例16-6:为UIKit视图控制器创建​​representable​​视图

import SwiftUIstruct MyViewController: UIViewControllerRepresentable {func makeUIViewController(context: Context) -> DetailViewController {let controller = DetailViewController()return controller}func updateUIViewController(_ uiViewController: UIViewControllerType, context: Context) {}
}

​UIViewControllerRepresentable​​​协议类似​​UIViewRepresentable​​​。我们必须定义一个实现此协议的结构体,然后添加创建和更新视图所需的方法。在本例中,我们仅定义了​​makeUIViewController()​​,因为只需要创建视图控制器的实例,来在屏幕上进行显示。以下的SwiftUI视图在用户点击按钮时在​​NavigationStack​​视图中加载该视图控制器。

图16-7:通过SwiftUI视图加载一个UIKit视图控制器

struct ContentView: View {var body: some View {NavigationStack {VStack {NavigationLink("Open UIKit View", destination: {MyViewController()}).buttonStyle(.borderedProminent)Spacer()}.padding()}}
}

以上视图包含一个加载​​MyViewController​​​结构体实例的​​NavigationLink​​视图,因而可以从初始视图导航至结构体所创建的视图控制器,和常规SwiftUI视图的操作一样。

图16-3:SwiftUI界面中的UIKit视图控制器

图16-3:SwiftUI界面中的UIKit视图控制器

✍️跟我一起做:选择屏幕顶部菜单中的File选项,新建一个文件。点击iOS版块中的Cocoa Touch Class图标,创建一个UIKit文件。在SubClass选项中选择​​UIViewController​​​。插入名称​​DetailViewController​​并点击Next进行保存。使用示例16-5中的代码更新​​DetailViewController​​​。用示例16-6中的代码创建一个名为​​MyViewController.swift​​​的文件。使用示例16-7中的代码更新​​ContentView​​视图。运行应用,点击按钮打开UIKit视图控制器。

代码请见:​​GitHub仓库​​

相关文章:

大师学SwiftUI第16章 - UIKit框架集成

其它相关内容请见​​虚拟现实(VR)/增强现实(AR)&visionOS开发学习笔记​​ SwiftUI是一套新框架&#xff0c;因此并没有包含我们构建专业应用所需的所有工具。这意味着我们会需要求助于UIKit&#xff08;移动设备&#xff09;和AppKit&#xff08;Mac电脑&#xff09;等原…...

7.docker运行redis容器

1.准备redis的配置文件 从上一篇运行MySQL容器我们知道&#xff0c;需要给容器挂载数据卷&#xff0c;来持久化数据和配置&#xff0c;相应的redis也不例外。这里我们以redis6.0.8为例来实际说明下。 1.1 查找redis的配置文件redis.conf 下面这个网址有各种版本的配置文件供…...

unity教程

前言 伴随游戏行业的兴起&#xff0c;unity引擎的使用越来越普遍&#xff0c;本文章主要记录博主本人入门unity的相关记录大部分依赖siki学院进行整理。12 一、认识unity引擎&#xff1f; 1、Unity相关信息&#xff1a; Unity的诞生&#xff1a;https://www.jianshu.com/p/550…...

未定义与 ‘double‘ 类型的输入参数相对应的函数 ‘Link‘

报错 检查对函数"Link"得调用中是否缺失参数或参数数据类型不正确。 未定义与"double"类型的输入参数相对应的函数"Link"。 问题描述 网上搜了搜一般说是toolbox没有下载导致的&#xff0c;相当于调用的包本地没有。 但是我看看了 Robotics…...

为什么Transformer模型中使用Layer Normalization(Layer Norm)而不是Batch Normalization(BN)

❤️觉得内容不错的话&#xff0c;欢迎点赞收藏加关注&#x1f60a;&#x1f60a;&#x1f60a;&#xff0c;后续会继续输入更多优质内容❤️ &#x1f449;有问题欢迎大家加关注私戳或者评论&#xff08;包括但不限于NLP算法相关&#xff0c;linux学习相关&#xff0c;读研读博…...

Vite - 配置 - 文件路径别名的配置

为什么要配置别名 别名的配置&#xff0c;主要作用是为了缩短代码中的导入路径。例如有如下的项目目录&#xff1a; project-name| -- src| -- a| --b| --c| --d| --e| -- abc.png| -- index.html| -- main.js如果想在 main.js 文件中使用 abc.png ,则使用的路径是 &#xff1…...

phpStorm Xdebug调试 加FireFox浏览器

步骤1&#xff1a; [Xdebug] zend_extension“D:\phpstudy_pro\Extensions\php\php5.4.45nts\ext\php_xdebug.dll” xdebug.collect_params1 xdebug.collect_return1 xdebug.remote_enableOn xdebug.remote_hostlocalhost xdebug.remote_port9001 xdebug.remote_handlerdbgp ;…...

多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测

多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测 目录 多维时序 | MATLAB实现PSO-BiGRU-Attention粒子群优化双向门控循环单元融合注意力机制的多变量时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 …...

linux配置固定ip(两种方法)

首先刚下载的vm&#xff0c;刚创建的虚拟机&#xff0c;肯定是需要配置ip的 其次以前我的每次都是设置自动ip&#xff0c;这样每次登录都会自动获取ip地址&#xff0c;并且每次的ip都不相同。 ~方法&#xff1a; 开机登陆后 1)Cd /etc/sysconfig/network-scripts 2)Vi ifcf…...

什么是缓存雪崩、击穿、穿透?

背景 数据一般是存储于数据库中&#xff0c;数据库中的数据都是存在磁盘上的&#xff0c;磁盘读写的速度相较于内存或者CPU中的寄存器来说是非常慢的了。 如果用户的请求都直接访问数据库的话&#xff0c;请求数量一上来&#xff0c;数据库很容易就崩溃了&#xff0c;所以为了…...

可以通过电脑远程控制安卓设备的软件

有些时候&#xff0c;我们需要用电脑远程控制安卓设备&#xff0c;比如远程维护门店设备、安卓系统的户外广告牌等等。我们来探索和比较几款允许用户通过电脑远程控制安卓设备的软件。 1、Splashtop Business Splashtop 是一种多功能远程访问解决方案&#xff0c;以其高性能流…...

HP惠普暗影精灵9笔记本电脑OMEN by HP Transcend 16英寸游戏本16-u0000原厂Windows11系统

惠普暗影9恢复出厂开箱状态&#xff0c;原装出厂Win11-22H2系统ISO镜像 下载链接&#xff1a;https://pan.baidu.com/s/17ftbBHEMFSEOw22tnYvPog?pwd91p1 提取码&#xff1a;91p1 适用型号&#xff1a;16-u0006TX、16-u0007TX、16-u0008TX、16-u0009TX、16-u0017TX 原厂系…...

vue2+elementUI 仿照SPC开发CPK分析工具

源码地址请访问 Vue CPK分析工具页面设计源码&#xff08;支持左右可拖拽和表格可编辑、复制粘贴&#xff09;仿照SPC开发-CSDN博客...

云ES使用集群限流插件(aliyun-qos)

aliyun-qos插件是阿里云Elasticsearch团队自研的插件,能够提高集群的稳定性。该插件能够实现集群级别的读写限流,在关键时刻对指定索引降级,将流量控制在合适范围内。例如当上游业务无法进行流量控制时,尤其对于读请求业务,可根据aliyun-qos插件设置的规则,按照业务的优先…...

2023.11.17 hadoop之HDFS进阶

目录 HDFS的机制 元数据简介 元数据存储流程:namenode 生成了多个edits文件和一个fsimage文件 edits和fsimage文件 SecondaryNameNode辅助NameNode的方式: HDFS的存储原理 写入数据原理: 发送写入请求,获取主节点同意,开始写入,写入完成 读取数据原理:发送读取请求,获取…...

如何在el-tree懒加载并且包含下级的情况下进行数据回显-01

在项目中做需求&#xff0c;遇到一个比较棘手的问题&#xff0c;el-tree懒加载在包含下级的时候&#xff0c;需要做回显&#xff0c;将选中的数据再次勾选上&#xff0c;在处理这个需求的时候有两点是比较困难的&#xff1a; el-tree是懒加载的&#xff0c;包含下级需要一层一…...

系列六、JVM的内存结构【栈】

一、产生背景 由于跨平台性的设计&#xff0c;Java的指令都是根据栈来设计的&#xff0c;不同平台的CPU架构不同&#xff0c;所以不能设计为基于寄存器的。 二、概述 栈也叫栈内存&#xff0c;主管Java程序的运行&#xff0c;是在线程创建时创建&#xff0c;线程销毁时销毁&…...

技巧篇:在Pycharm中配置集成Git

一、在Pycharm中配置集成Git 我们使用git需要先安装git工具&#xff0c;这里给出下载地址&#xff0c;下载后一路直接安装即可&#xff1a; https://git-for-windows.github.io/ 0. git中的一些常用词释义 Repository name&#xff1a; 仓库名称 Description(可选)&#xff1a;…...

Yolov5

Yolov5 Anchor &#xff11;&#xff0e;Anchor是啥&#xff1f; anchor字面意思是锚&#xff0c;是个把船固定的东东&#xff08;上图&#xff09;&#xff0c;anchor在计算机视觉中有锚点或锚框&#xff0c;目标检测中常出现的anchor box是锚框&#xff0c;表示固定的参考框…...

36、Flink 的 Formats 之Parquet 和 Orc Format

Flink 系列文章 1、Flink 部署、概念介绍、source、transformation、sink使用示例、四大基石介绍和示例等系列综合文章链接 13、Flink 的table api与sql的基本概念、通用api介绍及入门示例 14、Flink 的table api与sql之数据类型: 内置数据类型以及它们的属性 15、Flink 的ta…...

Cursor规则太多跑得慢?手把手教你优化.cursor配置,给VSCode插件‘减负’提速

Cursor性能优化实战&#xff1a;让智能编码助手重获流畅体验 当你的指尖在键盘上飞舞时&#xff0c;最令人沮丧的莫过于等待工具响应。作为深度集成AI能力的现代编码环境&#xff0c;Cursor在提供智能补全和代码建议的同时&#xff0c;也可能因为规则膨胀而逐渐变得迟缓。我曾见…...

从LTE到5G-Advanced:载波聚合(CA)技术演进全解析与网络工程师调试指南

从LTE到5G-Advanced&#xff1a;载波聚合技术深度演进与实战调试手册 当你在凌晨三点的基站机房盯着屏幕上跳动的KPI指标&#xff0c;突然发现某个5G小区下行速率始终无法突破800Mbps——这很可能是一个典型的载波聚合配置问题。作为网络优化工程师&#xff0c;我们每天都在与这…...

实战指南 | TSMaster 的 CAN UDS 诊断自动化流程与 BootLoader 刷写详解

1. TSMaster诊断控制台深度解析 诊断控制台是TSMaster进行UDS诊断的核心操作界面&#xff0c;相当于工程师与ECU对话的"翻译器"。我第一次接触这个界面时&#xff0c;被它清晰的四分区设计惊艳到了——就像汽车仪表盘把转速、车速、油量分区域显示一样直观。 服务命令…...

AsyncAPI通道管理终极指南:如何高效组织消息流的关键技巧

AsyncAPI通道管理终极指南&#xff1a;如何高效组织消息流的关键技巧 【免费下载链接】spec The AsyncAPI specification allows you to create machine-readable definitions of your asynchronous APIs. 项目地址: https://gitcode.com/gh_mirrors/spec/spec AsyncAPI…...

批量视频加图片水印工具使用指南

软件简介批量视频加图片水印工具是一款桌面端视频水印批量添加工具&#xff0c;支持单张/多张水印、九宫格固定位置、四种随机位置模式、大小和透明度调节、时间间隔水印等功能。核心功能功能说明单张水印所有视频使用同一张水印图片多张随机水印从水印文件夹中随机选择水印图片…...

在AutoDL上搞定nuScenes数据集:从解压到mmdetection3d初始化(含避坑指南)

在AutoDL云端高效部署nuScenes数据集&#xff1a;全流程解析与实战避坑指南 nuScenes作为自动驾驶领域最具挑战性的3D感知数据集之一&#xff0c;包含1000个复杂城市场景的多模态数据。但对于刚接触云端GPU服务器的研究者来说&#xff0c;从数据解压到环境配置的每一步都可能遇…...

贪心算法3(c++)

概念题目最短前缀题目描述 一个字符串的前缀是从该字符串的第一个字符起始的一个子串。例如carbon的字串是:cca,carcarb,carbo,和carbon。我们现在希望能用前缀来缩略的表示单词。例如,carbohydrate通常用carb来缩略表示&#xff0c;现在给你一组单词&#xff0c;要求你找到唯一…...

企业级邮件系统自建指南:从技术选型到生产部署

企业级邮件系统自建指南&#xff1a;从技术选型到生产部署 【免费下载链接】james-project James Project是一个用于电子邮件服务器的开源软件。适用于需要为其邮件基础设施提供强大和可靠的邮件传输代理的企业和组织。具有可扩展性、灵活性和易于使用的特点。 项目地址: htt…...

SUNFLOWER MATCH LAB在CSDN技术社区的分享:从部署到创新的完整旅程

SUNFLOWER MATCH LAB在CSDN技术社区的分享&#xff1a;从部署到创新的完整旅程 最近在CSDN上看到不少关于AI模型部署和应用的讨论&#xff0c;其中SUNFLOWER MATCH LAB这个项目引起了我的注意。它不是一个简单的模型调用工具&#xff0c;更像是一个围绕特定AI能力构建的完整实…...

Xilinx UltraScale GT收发器实战:从时钟配置到8B/10B编码的避坑指南

Xilinx UltraScale GT收发器实战&#xff1a;从时钟配置到8B/10B编码的避坑指南 在高速数字系统设计中&#xff0c;Xilinx UltraScale系列FPGA的GT收发器是实现多Gbps数据通信的核心组件。然而&#xff0c;许多工程师在实际部署时会遇到时钟配置混乱、弹性缓冲区溢出等棘手问题…...