在 MoonBit 中引入 Elm 架构:用简单原则打造健壮的 Web 应用
Elm 是一种纯函数式编程语言,专为构建前端 Web 应用程序而设计。它编译为 JavaScript,强调简洁性、性能和健壮性。
纯函数式的含义是函数没有副作用,这使得代码更易于理解和调试。通过强大的静态类型检查,Elm 确保应用程序不会抛出运行时异常,从而提供更高的可靠性。Elm 架构强制执行单向数据流,使状态管理变得可预测且简单。
受 Elm 的启发,我们使用 MoonBit 开发了一个名为 Rabbit-TEA 的前端 Web 框架。
为什么选择 MoonBit?
函数式编程的兴起也推动了 React 从基于类的范式向 React Hooks 的演进。在 JavaScript 中已经有了一些函数式框架,那么为什么还需要 MoonBit?
尽管 JavaScript 生态中已经有一些支持函数式风格的框架和库,但 JavaScript 本身缺乏诸如模式匹配等特性,这使得编码体验不够理想。以下是 MoonBit 和 JavaScript 等效代码的对比:
let x = match y {a => value1b => value2c => value3
}
f(x)
var x = null;
if (y == a) { x = value1 }
else if (y == b) { x = value2 }
else { x = value3 }
// x 可能会在这里被意外修改!
return f(x);// 或者
const x = y == a ? value1 : (y == b ? value2 : value3);
return f(x);
以下是另一个 MoonBit 例子。函数 validate_utf8
检查输入的 bytes
是否是有效的 UTF-8 序列:
fn validate_utf8(bytes : Bytes) -> Bool {loop bytes {[0x00..=0x7F, ..xs]| [0xC0..=0xDF, 0x80..=0xBF, ..xs]| [0xE0..=0xEF, 0x80..=0xBF, 0x80..=0xBF, ..xs]| [0xF0..=0xF7, 0x80..=0xBF, 0x80..=0xBF, 0x80..=0xBF, ..xs] => continue xs[_, ..] => false[] => true}
}
MoonBit 是一种面向表达式的语言,变量默认是不可变的。它完全支持模式匹配这一在函数式编程语言中流行已久的特性,确保生成高效的 JavaScript 代码。此外,MoonBit 编译器支持多种后端,包括 JavaScript、WebAssembly 和 native 后端。
Rabbit-TEA 如何工作
TEA 架构非常的简单,它由 Model、View、Update 组成:
- Model:App 的状态,由不可变的数据结构组成。
- View:App 的视图,声明了如何将 Model 显示为 HTML、用户的操作触发什么消息。
- Update:App 的逻辑,处理用户触发的消息,将旧的 Model 转换为新的 Model。
下面是一个完整的计数器程序:
typealias Model = Int
let model = 0enum Msg {IncrementDecrement
}fn update(msg : Msg, model : Model) -> (Command[Msg], Model) {match msg {Increment => (none(), model + 1)Decrement => (none(), model - 1)}
}fn view(model : Model) -> Html[Msg] {div([h1([text(model.to_string())]),button(click=Msg::Increment, [text("+")]), // button1button(click=Msg::Decrement, [text("-")]), // button2])
}fn main {@tea.startup(model~, update~, view~)
}
这个程序除了定义 Model、Update 和 View,还定义了 Msg
类型。Msg
类型类似于事件。当用户点击增加按钮时,会触发运行时向 update
函数发送 Increment
消息以及当前的 model
。在 update
函数中,并不会直接修改 model
,而是基于旧值创建一个新的 model
并返回。整个函数是无副作用的。最后,新的 model
将通过 view
函数渲染为 HTML。
update
函数还返回一个 Cmd
类型,它表示一个未执行的操作。我们稍后会对此进行介绍。
声明式视图: HTML EDSL
EDSL 即内嵌领域特定语言,是通过利用编程语言已有的语法结构设计的 DSL,不需要额外的预处理和编译步骤。
Rabbit-TEA 使用 MoonBit 的 labeled argument 语法糖定义了一组辅助函数,这些 HTML 函数尽可能地通过类型来提示用户参数的作用, 避免滥用模糊和令人困惑的 String
类型。它们使用起来就像:
fn view() -> Html[Msg] {div([h1([text("hello MoonBit")]),p([text("build robust app")]),a(href="moonbitlang.com", target=Blank, [text("try it")]),])
}
以标签 <a>
为例子,它的辅助函数的类型定义如下。除了唯一的 childrens
参数,其他所有参数都是 labeled argument,labeled argument 根据需要可以声明为可选的,或者设置缺省值。
pub(all) enum Target {SelfBlank
}pub fn a[M](style~ : Array[String] = [],id? : String,class? : String,href~ : String,target~ : Target = Self,childrens : Array[Html[M]]
) -> Html[M]
在未来,我们将考虑在 MoonBit 中引入类似 JSX 的拓展语法,使视图的代码拥有更好的可读性。
消息与模式匹配
TEA 架构充分利用了模式匹配和带标签联合类型,用户定义的 Msg 可以视情况带上额外的数据,这使得 update
处理消息时更加安全和灵活。
例如,对于一个 input
元素,假如希望在用户输入的同时,处理它的值并将结果显示在另一个元素 p
中。我们可以在 Msg
类型中增加一个带 String
数据的 enum constructor,GotInputChange
。GotInputChange
的类型和 input
中 change
参数的类型都是 (String) -> Msg
。用户在修改 input
的值时,input
的值会被打包进 GotInputChange
并发送给 update
函数处理。
enum Msg {GotInputChange(String)
}fn update(msg : Msg, model : String) -> (Cmd[Msg], String) {match msg {GotInputChange(value) => {...(none(), value)}}
}fn view(model : String) -> Html[Msg] {div([p([text(model)])input(input_type=Text, value=model, change=Msg::GotInputChange),])
}
依靠 MoonBit 的穷尽性检查,下次 Msg
增加新的 enum constructor 时,IDE 将会提示相关的 pattern matching 需要处理新的 case,减少运行时错误。
管理副作用
Rabbit-TEA 与 Elm 一样使用 Cmd 类型管理函数的副作用,也就是修改外部状态、与外部系统交互的操作。
例如,请求浏览器记录 URL History、滚动浏览器视图、向服务器发送 JSON 请求并处理返回的数据。
所有的这些操作被封装为 Cmd。Cmd 代表一个未执行的动作,只有当它被 Update 函数的返回时,才会被 Rabbit-TEA 的 runtime 触发执行。
下面的例子演示了在用户点击相关的卡片时,先将界面转换为加载状态,然后通过 http 请求加载卡片的内容。
@http.get
返回了一个 Cmd,而它的第一个参数是请求的 URL,第二个参数声明期望的数据格式和处理响应的方法。
enum Msg {GotCardText(Result[String, String])ClickCard(Int)
}pub fn update(msg : Msg, model : Model) -> (Cmd[Msg], Model) {match msg {ClickCard(id) => (@http.get("/api/card/\{id}", expect=Text(GotCardText)), Loading)GotCardText(Ok(text)) => (none(), Editing(text))GotCardText(Err(_)) => (none(), NotFound)}
}
这样一来,服务器响应请求时将触发 GotCardText
这个消息。它带上了一个 Result[String,String]
类型,
这提示我们对响应进行分类讨论:当成功返回 card text 时,我们更新 model
以便 view
向用户展示内容;当请求失败时,我们跳转到 NotFound
页面。
Cmd
模式鼓励用户避免在一个函数中交替地触发副作用和更新 Model,从而防止 update 和 view 同时存在两个 Model 的数据来源,保证程序状态的一致性。
与 JS 世界交互
上面提到的这些 Cmd 并非经过框架特殊处理,而是基于 Cmd 的接口实现的。你也可以定义自己的 Cmd 来让 Rabbit-TEA 和外部的 JS 世界交互。MoonBit 允许以内联的形式书写 JS 代码,并将它和函数定义绑定。这很简单:
extern "js" fn set_timeout(f : () -> Unit, ms : Int) = "(f,ms) => setTimeout(f, ms)"
set_timeout
绑定了 JS 的函数 setTimeout
。参数 f 表示将要触发的动作,ms
表示触发前等待的时间。
我们定义一个函数 delay
。它接受的参数 msg
表示将要触发的消息,ms
作用同上,最终返回 Cmd
。
fn delay[M](msg : M, ms : Int) -> Cmd[M] {Cmd(fn(events){set_timeout(fn(){ events.trigger_update(msg) }, ms)})
}
Cmd
类型的值可以通过同名的 Cmd
构造器创建,它是一个未执行的动作的包装。在这个包装中,我们让 set_timeout
在合适的时机调用 events.trigger_update(msg)
,让运行时使用 msg
和当前的 model
调用 update
函数。
我们可以使用这个 delay
函数,在用户点击按钮的5秒后显示一段文本:
enum Msg {Timeout(String)SetTime
}fn update(msg : Msg, model : String) -> (Cmd[Msg], String) {match msg {Timeout(tips) => (none(), tips)SetTime => (delay(Timeout("You clicked the button 5s ago."), 5000), "")}
}fn view(model : String) -> Html[Msg] {div([text(model),button(click=Msg::SetTime, [text("show tips after 5s")]),])
}
不过,在编写内联 JS 时需要小心运行时错误。我们计划在 Rabbit-TEA 中提供更丰富的 Cmd,用户一般不需要自己编写这些绑定。
轻量的运行时
我们开发 Rabbit-TEA 时并没有注意过编译生成的 js 文件的大小。得益于 MoonBit 的全局 DCE 优化,我们发现 Rabbit-TEA 在带上 virtual dom 和 MoonBit 的标准库的同时,一个计数器应用生成的代码仅 33KB(经过 minify,未 gzip),小于 vue 的 50~60kb。
结论
在实现 Rabbit-TEA 的同时,我们还用它重写 MoonBit 的包管理网站 mooncakes.io。目前 mooncakes.io 和 Rabbit-TEA 还不够完善。我们希望随着 mooncakes.io 的实际应用,Rabbit-TEA 也能够成为成熟可靠的 UI 框架。我们未来也计划探索 TEA 架构的更多可能性,例如 native 后端支持、服务端渲染、时间旅行调试等等令人兴奋的特性。
如果你对这个框架和 MoonBit 感兴趣,也欢迎参与到 MoonBit 的社区当中。
实习机会
我们也开放了一个前端实习岗位,负责 MoonBit 前端框架 Rabbit-TEA 和周边工具的研发。如果你了解前端框架中的 Virtual DOM、Server Side Rendering、状态管理等机制,了解函数式编程范式、The Elm Architecture 或者 React + Redux,欢迎复制打开链接或扫描下方二维码投递简历。
https://idea.zhiye.com/intern/detail?jobAdId=c2b87413-5ab5-40d3-93c7-a65ca48300e04
下一步
- 下载 MoonBit
- MoonBit 初学者之旅
- MoonBit 语言导览
- 查看 MoonBit 文档
相关文章:
在 MoonBit 中引入 Elm 架构:用简单原则打造健壮的 Web 应用
Elm 是一种纯函数式编程语言,专为构建前端 Web 应用程序而设计。它编译为 JavaScript,强调简洁性、性能和健壮性。 纯函数式的含义是函数没有副作用,这使得代码更易于理解和调试。通过强大的静态类型检查,Elm 确保应用程序不会抛…...
虚幻基础:ue引擎的碰撞
文章目录 碰撞:碰撞体间 运动后 产生碰撞的行为——由引擎负责,并向各自发送事件忽略重叠阻挡 碰撞体类型模式纯查询:不清楚具体作用可以阻挡 actor碰撞(武器:刀/子弹)子组件可以产生阻挡 角色的碰撞只有根组件可以阻挡࿰…...
「电商玩法」AI自动创作系统源码:商品图+视频+营销文案一键生成
—零代码搭建智能内容工厂,1人日更1000条爆款素材 电商行业核心痛点 1. 内容产能不足 中小商家无力承担专业摄影/剪辑,商品图质量差→转化率<1%热点借势慢:竞品已开始推“淄博烧烤同款”,你的素材还在拍摄中 2. 成本居高不下…...
图形变换算法
一、学习目的 (1)掌握多面体的存储方法。 (2)掌握图形的几何变换及投影变换。 (3)掌握三维形体不同投影方法的投影图的生成原理。 (4)掌握多面体投影图绘制的编程方法。 二、学…...
no such window: target window already closed的解决方法
我在使用selenium 切换窗口的时候,由于不小心关闭了一个窗口,运行的时候就遇到这样的警告: no such window: target window already closed 具体的问题展示: 这个问题表示:当前的页面被关闭了,selenium 找…...
vue常见错误
1、 Cant resolve vant/lib/index.less 1. 未正确安装 Vant 首先,确保你已经正确安装了 Vant。可以通过以下命令来安装: npm install vant --save 或者使用 yarn: yarn add vant 2. LESS 加载器未配置 如果你在项目中使用了 Vant 的 L…...
chrome中的copy xpath 与copy full xpath的区别
学过测试或者爬虫的,都感觉获取网页元素,使用xpath最方便 但其中有一些细节可能会使你摸不清头脑 比如有时候copy xpath会定位不准确,而使用copy full xpath就可以定位 1、copy xpath(相对路径定位) 优点ÿ…...
【Docker】运行错误提示 unknown shorthand flag: ‘d‘ in -d ----详细解决方法
使用docker拉取Dify的时候遇到错误 错误提示 unknown shorthand flag: d in -dUsage: docker [OPTIONS] COMMAND [ARG...]错误原因解析 出现 unknown shorthand flag: d in -d 的根本原因是 Docker 命令格式与当前版本不兼容,具体分为以下两种情况: 新…...
VS Code 安装及常用插件
一、VS Code下载与安装 1、概述 Visual Studio Code简称VS Code,是一款功能强大的代码编辑器,与IDE(集成开发环境)不同,VS Code需要安装平台相应的编译器和语言的扩展。 IDE:是用于提供程序开发环境的应…...
iptables防火墙
目录 一 Linux防火墙基础 1 iptables的表,链结构 (1)规则表 filter 表 nat 表 mangle 表 raw 表 (2)规则链 2 数据包过滤的匹配流程 (1)规则表之间的顺序 (2)…...
【JavaWeb】详细讲解 HTTP 协议
文章目录 一、HTTP简介1.1 概念1.2 特点 二、协议2.1 HTTP-请求协议(1)GET方式(2)POST方式(3)GET和POST的区别: 2.2 HTTP-响应协议(1)格式(2)响应…...
非阻塞I/O操作
非阻塞I/O操作是一种I/O操作模式,在这种模式下,应用程序在发出I/O请求后不会立即等待操作完成,而是继续执行其他任务。当I/O操作完成或可以进行时,系统会通知应用程序。这种操作模式可以提高程序的效率和响应能力,因为…...
Redis面试问题详解2
Redis面试问题详解2 一、分布式锁 分布式锁主要用于解决多服务器之间的并发问题。Redis通过SETNX命令实现分布式锁,确保同一时间只有一个线程可以获取锁。 1. 基本实现 获取锁 使用SETNX命令设置锁,并设置一个过期时间,避免死锁。 Stri…...
【软件测试】性能测试概念篇
1. 性能测试的定义 性能测试是通过模拟真实用户行为、系统负载或极端条件,评估软件系统在特定场景下的响应能力、稳定性、资源消耗及扩展性的过程。其核心目标是: 验证系统容量:确保系统在预期负载下(如…...
在Pycharm配置stable diffusion环境(使用conda虚拟环境)
自己配环境的时候也没个指南,少安装包或者包之间版本冲突是再按正常不过的事了,真的令人不胜其烦。 下面记录一下自己在conda虚拟环境配置stable diffusion的代码环境,希望能帮大家少踩几个坑。 虚拟环境配置 默认你已经安装了annaconda&am…...
Uniapp微信小程序:轻松获取用户头像和昵称
参考文献:Uniapp微信小程序:轻松获取用户头像和昵称-百度开发者中心 (baidu.com) uni.login({ provider: weixin, success: function (loginRes) { console.log(loginRes.authResult); // 打印登录凭证 // 使用登录凭证获取用户信息 uni.getUserInfo({ …...
Qt核心知识总结
Qt核心知识总结 Qt 是一个功能强大、跨平台的 C 应用程序开发框架,广泛应用于图形用户界面(GUI)应用程序的开发,同时也支持非 GUI 应用程序的开发。本文将从入门到精通的角度,详细解析 Qt 的核心知识点,帮…...
Doris的向量化执行如何支撑分布式架构和复杂查询
Doris 的向量化执行能力与其 分布式架构 和 复杂查询优化 深度结合,通过 批处理 列式计算 分布式调度 的协同设计,解决传统分布式数据库在复杂查询场景下的性能瓶颈。以下是具体原理展开: 一、向量化如何适配分布式架构? Doris…...
无源蓝牙技术与传统RFID(射频识别)对比
百事可乐的无源蓝牙技术与传统RFID(射频识别)虽然都属无线通信技术,但在工作原理、应用场景和技术性能上存在显著差异。以下是深度对比分析: 一、核心技术差异 维度无源蓝牙(如百事方案)R…...
Operator 开发入门系列(一):Hello World
背景 我们公司最近计划将产品迁移到 Kubernetes 环境。 为了更好地管理和自动化我们的应用程序,我们决定使用 Kubernetes Operator。 本系列博客将记录我们学习和开发 Operator 的过程,希望能帮助更多的人入门 Operator 开发。 目标读者 对 Kubernete…...
Immich图库本地部署与远程管理:打造你的专属照片云服务
文章目录 前言1.关于Immich2.安装Docker3.本地部署Immich4.Immich体验5.安装cpolar内网穿透6.创建远程链接公网地址7.使用固定公网地址远程访问 前言 小伙伴们,你们好呀!今天要给大家带来一个超炫的技能——让你家的电脑秒变私人云相册,并且…...
OpenShift AI + DeepSeek+IBM CP4BA:实现AI业务自动化中台
一、OpenShift AI DeepSeek 1、什么是 OpenShift AI? OpenShift AI(前身为 Red Hat Open Data Hub)是 Red Hat OpenShift 平台上的一套 AI/ML 工具集,基于 Kubernetes,为企业提供可扩展的机器学习平台,可…...
游戏引擎学习第229天
仓库:https://gitee.com/mrxiao_com/2d_game_5 回顾上次内容并介绍今天的主题 上次留下的是一个非常简单的任务,至少第一步是非常简单的。我们需要在渲染器中加入排序功能,这样我们的精灵(sprites)才能以正确的顺序显示。为此我…...
genhtml 工具使用说明
genhtml 工具使用说明 简介 genhtml 是一个用于生成 HTML 格式代码覆盖率报告的工具,基于 lcov 生成的覆盖数据文件(如 lcov.info)生成交互式 HTML 报告。它支持多种自定义选项,可控制报告内容、样式、输出格式等。 基本用法 ge…...
C++11_2
文章目录 前言一、新的类功能——新的默认成员函数1. 编译器默认生成的移动构造与移动赋值2. 手动提供移动构造或移动赋值的影响 二、可变参数模板1. 多参数1. 可变参数模板的基础 2. 可变参数模板的展开(1)递归展开参数包(2)使用…...
HTML语义化与无障碍设计
HTML 语义化与无障碍设计:构建包容且高效的网页体验 引言 在我的前端开发学习旅程中,起初将 HTML 仅视为页面布局的工具,大量使用无语义的 <div> 和 <span>。直到在一篇技术博客当中了解到,作者在一次团队项目中&am…...
Ubuntu多用户VNC远程桌面环境搭建:从零开始的完整指南
引言: 在当今远程工作盛行的时代,搭建一个安全、高效的多用户远程桌面环境变得越来越重要。本文将为您提供一个从零开始的完整指南,教您如何在Ubuntu系统上搭建多用户VNC远程桌面环境。无论您是系统管理员、开发团队负责人,还是想要为家庭成员提供远程访问的技术爱好者,这…...
Electron 中引入MessageChannel 大大缩短不同渲染进程和 Webview 各组件 1o1的通信链路
背景 在 electron 开发中,也不可避免地遇到端到端的通信问题,Electron 已经内置一些通信 API,但是实际用下来会发现,在引入 Webview 之后,通信链路会很长,参考 利用本地 Express Web 服务解决复杂的 Elec…...
Vscode开发Vue项目NodeJs启动报错处理
文章目录 背景一、npm启动报错报错信息定位原因处理方案第一步、下载安装高版本 二、node 无法识别报错信息处理方案定位原因第一步、检测环境变量第二步、重新开启界面 背景 使用Vscode开发Vue项目,使用到NodeJs,记录出现的问题及处理方案,…...
AI agents系列之AI工作流和AI智能体对比
在人工智能(AI)快速发展的今天,理解AI工作流和AI智能体之间的区别对于有效利用这些技术至关重要。本文将深入探讨AI工作流的类型,解析AI智能体的概念,并重点比较二者的关键差异。 1. 智能体 vs 工作流 关于“智能体”的定义众说纷纭。有些客户将其视为完全自主的系统,能…...
如何恢复极狐GitLab?
极狐GitLab 是 GitLab 在中国的发行版,关于中文参考文档和资料有: 极狐GitLab 中文文档极狐GitLab 中文论坛极狐GitLab 官网 恢复极狐GitLab (BASIC SELF) 极狐GitLab 提供了一个命令行界面来恢复整个安装,足够灵活以满足您的需求。 恢复…...
基于X86/Nvidia+FPGA大模型具身智能机器人控制器解决方案,同时拥有算力与实时的便利
2025年成为人形机器人产业化元年,行业已突破早期实验室研发阶段,进入"场景验证量产爬坡"新周期,预计2031年具身智能市场规模有望突破万亿元。这一进程的背后,是硬件算力、实时控制、环境适应等底层技术的系统性突破——…...
MATLAB项目实战(一)
题目: 某公司有6个建筑工地要开工,每个工地的位置(用平面坐标系a,b表示,距离单位:km)及水泥日用量d(t)由下表给出.目前有两个临时料场位于A(5,1),B(2,7),日储…...
PyCharm Flask 使用 Tailwind CSS 配置
使用 Tailwind CSS 步骤 1:初始化项目 在 PyCharm 终端运行:npm init -y安装 Tailwind CSS:npm install -D tailwindcss postcss autoprefixer初始化 Tailwind 配置文件:npx tailwindcss init这会生成 tailwind.config.js。 步…...
Mybtis和Mybatis-Plus区别
MyBatis 和 MyBatis-Plus 是 Java 中常用的持久层框架,MyBatis-Plus 是在 MyBatis 基础上增强的工具包,让开发更便捷、高效。下面是两者主要的区别: ✅ 核心区别总结: 特性MyBatisMyBatis-Plus配置复杂度需要手写大量 XML 或注解…...
《Learning Langchain》阅读笔记2-基于 Gemini 的 Langchain PromptTemplate 实现方式
本文将使用Gemini实现《Learning Langchain》中的PromptTemplate 实现方式,替代书中的调用openai API,白嫖太香了! 调试步骤 我们首先还是先在本地调试是否可以调用Gemini API: import getpass import osif "GOOGLE_API_K…...
LVS+keepalived搭建高可用架构
背景:最近在搭建LVSkeepalived的高可用架构,中间遇到了一些坑比较让人头疼,此处重要就安装部署的步骤进行记录,特别是遇到坑进行说明,希望能对有需要的同学提供给帮助! 坑点1: 在部署LVSkeepalived并且使用…...
【天梯赛练习】L2-035 完全二叉树的层序遍历
后序遍历转层序遍历 后序遍历:左——右——根层序遍历:数组形式存储的完全二叉树的顺序遍历序列其实就正好是其层序遍历序列。 子树根若是 i d id id,左子树 i d ∗ 2 id*2 id∗2,右子树 2 ∗ i d 1 2*id1 2∗id1 所以就是dfs递…...
2025.4.20机器学习笔记:文献阅读
2025.4.20周报 题目信息摘要创新点网络架构实验生成性能对比预测性能对比 结论不足以及展望 题目信息 题目: A novel flood forecasting model based on TimeGAN for data-sparse basins期刊: Stochastic Environmental Research and Risk Assessment作…...
Leetcode 3359. 查找最大元素不超过 K 的有序子矩阵【Plus题】
1.题目基本信息 1.1.题目描述 给定一个大小为 m x n 的二维矩阵 grid。同时给定一个 非负整数 k。 返回满足下列条件的 grid 的子矩阵数量: 子矩阵中最大的元素 小于等于 k。 子矩阵的每一行都以 非递增 顺序排序。 矩阵的子矩阵 (x1, y1, x2, y2) 是通过选择…...
Redis面试——事务
一、Redis原子性是什么? (1)单个命令的原子性 原子性是指一组操作,要么全部执行成功,要么全部失败。Redis 中的单个命令是天然原子性的,因为 Redis 的命令执行采用单线程模型,同一时间只会执行…...
【远程管理绿联NAS】家庭云存储无公网IP解决方案:绿联NAS安装内网穿透
文章目录 前言1. 开启ssh服务2. ssh连接3. 安装cpolar内网穿透4. 配置绿联NAS公网地址 前言 大家好,今天要带给大家一个超级酷炫的技能——如何让绿联NAS秒变‘千里眼’,通过简单的几步操作就能轻松实现内网穿透。想象一下,无论你身处何地&a…...
AI写程序:用 AI 实现一个递归批量转化 GBK/GB2312 转 UTF-8 工具:轻松解决文本编码转换难题
用 AI 实现一个递归批量转化 GBK/GB2312 转 UTF-8 工具 在处理历史文件或与不同系统交互时,我们经常会遇到 GBK 或 GB2312 编码的文本文件。虽然现在 UTF-8 是主流,但手动转换这些旧编码文件既繁琐又容易出错。为了解决这个问题,我开发了一个…...
首席人工智能官(Chief Artificial Intelligence Officer,CAIO)的详细解析
以下是**首席人工智能官(Chief Artificial Intelligence Officer,CAIO)**的详细解析: 1. 职责与核心职能 制定AI战略 制定公司AI技术的长期战略,明确AI在业务中的应用场景和优先级,推动AI与核心业务的深度…...
uview1.0 tabs组件放到u-popup中在微信小程序中滑块样式错乱
解决思路 重新计算布局信息:在弹窗显示后重新调用 init 方法来计算组件的布局信息。使用 nextTick:保证在视图更新之后再进行布局信息的计算。 <u-tabs ref"tabsRef" ></u-tabs> makeClick(){this.makeShowtruethis.$nextTick…...
私人笔记:动手学大模型应用开发llm-universe项目环境创建
项目代码:datawhalechina/llm-universe: 本项目是一个面向小白开发者的大模型应用开发教程,在线阅读地址:https://datawhalechina.github.io/llm-universe/ 项目书:动手学大模型应用开发 一、初始化项目 uv init llm-universe-te…...
基于Django框架的图书索引智能排序系统设计与实现(源码+lw+部署文档+讲解),源码可白嫖!
摘要 时代在飞速进步,每个行业都在努力发展现在先进技术,通过这些先进的技术来提高自己的水平和优势,图书管理系统当然不能排除在外。图书索引智能排序系统是在实际应用和软件工程的开发原理之上,运用Python语言以及Django框架进…...
网络类型学习
网络类型的分类依据-----基于二层(数据链路层)使用的协议不同而导致数据包的封装方式不同,工作方式也不同。 OSPF协议根据链路层协议类型将网络分为四种类型:广播型网络(BMA)、非广播多路访问(…...
ubuntu24.04离线安装deb格式的mysql-community-8.4.4
1,下载解压 参考: https://blog.csdn.net/2202_76101487/article/details/145967039 下载: wget https://cdn.mysql.com//Downloads/MySQL-8.4/mysql-server_8.4.4-1ubuntu24.04_amd64.deb-bundle.tar 建议个目录mysql8然后把安装包移过去&…...
电控---printf重定向输出
在嵌入式系统开发中,printf 重定向输出是将标准输出(stdout)从默认设备(如主机终端)重新映射到嵌入式设备的特定硬件接口(如串口、LCD、USB等)的过程。 一、核心原理:标准IO库的底层…...