Spring AI 实战:第三章、Spring AI结构化输出之告别杂乱无章
引言:当程序员遇上剧荒
“周末看什么?”
这个看似简单的问题,往往能让我们在各大影视平台间反复横跳半小时,最后无奈选择重刷《老友记》。本期让我们用技术解决这个"世纪难题":让大模型成为你的私人影视推荐官
一、输出结构化 - 轻松拿捏数据
1.1 原始输出的烦恼
简单示例:查一位idol主演的电影
@GetMapping("/idol/movie")public String idolMovies(@RequestParam(value = "idol", defaultValue = "刘德华") String idol) {String content = chatClient.prompt().user(u -> {u.text("帮我找五部{idol}主演的电影").param("idol", idol);}).call().content();log.info(content);return content;}
结果输出:
以下是五部由刘德华主演的经典电影推荐,涵盖不同风格和年代的作品:
《无间道》(2002)
类型:警匪/悬疑
角色:刘建明(反派卧底警察)
亮点:香港警匪片巅峰之作,刘德华与梁朝伟的双雄对决,剧情反转深刻。
《天若有情》(1990)
类型:爱情/动作
角色:华弟(街头浪子)
亮点:经典悲情爱情片,刘德华骑摩托载吴倩莲的画面成为影史名场面。
《暗战》(1999)
类型:犯罪/剧情
角色:张彼得(高智商劫匪)
亮点:刘德华凭此片首获香港金像奖影帝,与刘青云的斗智戏码精彩。
《桃姐》(2011)
类型:家庭/温情
角色:罗杰(少爷)
亮点:现实主义题材,刘德华与叶德娴的细腻演出,斩获多项大奖。
《拆弹专家2》(2020)
类型:动作/犯罪
角色:潘乘风(残障拆弹专家)
亮点:高燃爆炸场面与复杂角色塑造,近年港产动作片代表作。
其他备选:
《赌神》(1989)饰演陈刀仔(喜剧/赌片)
《十面埋伏》(2004)饰演刘捕头(武侠/爱情)
如需特定类型或年代的推荐,可以进一步补充说明!
问题来了:大模型默认输出自然语言(如纯文本),需额外解析(正则表达式、NLP 处理)才能被程序使用,容易出错
- 不同模型输出格式不一致
- 返回格式不稳定,难以用程序解析
- 无法直接映射到Java对象
1.2 结构化的优势
直接输出 JSON、XML、YAML 等格式,程序可无缝解析,无需复杂后处理,以结构化的输出才能更精准的控制整个流程以及和用户的交互形态,提升产品用户体验
{"name": "无间道","releaseDate": "2002-12-12","directorName": "刘伟强","desc": "卧底警察深入黑帮调查的故事"
}
Spring AI通过在提示词中添加格式化指令要求大模型按特定格式做内容返回,在拿到大模型输出数据后通过内置转换器的能力做结构化输出。
二、Spring AI结构化输出实战
2.1 BeanOutputConverter:让AI输出Java对象
实体定义:定义一个对象 Film
来标识电影信息实体
record Film(@JsonPropertyDescription("电影名称") String name,@JsonPropertyDescription("上映时间") String releaseDate,@JsonPropertyDescription("导演名称") String directorName,@JsonPropertyDescription("电影简介") String desc) {}
结构转换:使用BeanOutputConverter转化模型返回数据
@GetMapping("/film")
public Film queryFilm(@RequestParam(value = "filmName", defaultValue = "无间道") String filmName) {BeanOutputConverter<Film> beanOutputConverter = new BeanOutputConverter<>(new ParameterizedTypeReference<Film>() {});String content = chatClient.prompt().user(u -> {u.text("""从豆瓣上查询影视作品{filmName}的信息 {format}""").params(Map.of("filmName", filmName, "format", beanOutputConverter.getFormat()));}).call().content();return beanOutputConverter.convert(content);
}
输出如下, 很显然按照指定实体对象做结果的输出
{
“name”: “无间道”,
“releaseDate”: “2002-12-12”,
“directorName”: “刘伟强、麦兆辉”,
“desc”: “《无间道》是一部2002年上映的香港警匪电影,讲述了警方和黑帮互相派遣卧底的故事,展现了复杂的身份认同和道德困境。”
}
整体流程:
- 通过
<font style="color:rgba(0, 0, 0, 0.9);">BeanOutputConverter<T></font>
的泛型约定返回结果的实体对象 - 在提示词后面追加
**<font style="color:rgba(0, 0, 0, 0.9);">{format}</font>**
占位符,该占位符对应的值为<font style="color:rgba(0, 0, 0, 0.9);">beanOutputConverter.getFormat()</font>
- 调用
<font style="color:rgba(0, 0, 0, 0.9);">beanOutputConverter.convert(content)</font>
处理大模型返回数据,得到返回实体
<font style="color:rgba(0, 0, 0, 0.9);">BeanOutputConverter</font>
的泛型除了可以指定一个普通实体对象外,还可以指定List和Map。
List示例
@GetMapping("/idol/film/bean/list/format")public List<Film> idolFilmsBeanListFormat(@RequestParam(value = "idol", defaultValue = "刘德华") String idol) {BeanOutputConverter<List<Film>> beanOutputConverter = new BeanOutputConverter<>(new ParameterizedTypeReference<List<Film>>() {});String content = chatClient.prompt().user(u -> {u.text("""帮我找五部{idol}主演的电影{format}""").params(Map.of("idol", idol, "format", beanOutputConverter.getFormat()));}).call().content();List<Film> filmList = beanOutputConverter.convert(content);log.info("BeanOutputConverter#filmList,size={},resp={}", filmList.size(), filmList);return filmList;}
输出:
[
{
"name": "无间道","releaseDate": "2002-12-12","directorName": "刘伟强","desc": "卧底警察深入黑帮调查的故事"
},
{
"name": "赌神","releaseDate": "1989-12-14","directorName": "王晶","desc": "赌神高进的传奇故事"
},
{
"name": "暗战","releaseDate": "1999-09-23","directorName": "陈木胜","desc": "警察与盗贼之间的斗智斗勇"
},
{
"name": "天若有情","releaseDate": "1990-06-14","directorName": "杜琪峰","desc": "关于时间与爱情的科幻电影"
},
{
"name": "拆弹专家","releaseDate": "2017-04-28","directorName": "邱礼涛","desc": "警察与罪犯之间的心理博弈"
}
]
Map示例: 目前在Map中嵌套复杂类型还不支持,因此Map返回用的是Object而不是Film
@GetMapping("/bean/map/format")
public Map<String, Object> beanMapFormat(@RequestParam(value = "style", defaultValue = "华语流行") String style) {BeanOutputConverter<Map<String, Object>> beanOutputConverter = new BeanOutputConverter<>(new ParameterizedTypeReference<Map<String,Object>>() {});String content = chatClient.prompt().user(u -> {u.text("""帮我找五部{style}的电影,以电影名为分组键,值为电影信息,电影信息需要包含电影名称、上映时间、导演名、电影简介等内容{format}""").params(Map.of("style", style, "format", beanOutputConverter.getFormat()));}).call().content();Map<String, Object> filmMap = beanOutputConverter.convert(content);log.info("BeanOutputConverter#filmMap,size={},resp={}", filmMap.size(), filmMap);return filmMap;
}
输出:
{
“霸王别姬”: {
"电影名称": "霸王别姬","上映时间": "1993年","导演名": "陈凯歌","电影简介": "影片围绕两位京剧伶人半个世纪的悲欢离合,展现了对传统文化、人的生存状态及人性的思考与领悟。"
},
“无间道”: {
"电影名称": "无间道","上映时间": "2002年","导演名": "刘伟强、麦兆辉","电影简介": "讲述的是两个身份混乱的男人分别为警方和黑社会的卧底,经过一场激烈的角斗,他们决心要寻回自己的故事。"
},
“卧虎藏龙”: {
"电影名称": "卧虎藏龙","上映时间": "2000年","导演名": "李安","电影简介": "一代大侠李慕白有退出江湖之意,托付红颜知己俞秀莲将自己的青冥剑带到京城,作为礼物送给贝勒爷收藏。"
},
“让子弹飞”: {
"电影名称": "让子弹飞","上映时间": "2010年","导演名": "姜文","电影简介": "讲述了悍匪张牧之摇身一变化名清官“马邦德”上任鹅城县长,并与镇守鹅城的恶霸黄四郎展开一场激烈争斗的故事。"
},
“大话西游之大圣娶亲”: {
"电影名称": "大话西游之大圣娶亲","上映时间": "1995年","导演名": "刘镇伟","电影简介": "至尊宝被月光宝盒带回到五百年前,遇见紫霞仙子,被对方打上烙印成为对方的人,并发觉自己已变成孙悟空。"
}
}
2.2 优雅的entity()写法
大模型输出转换为实体对象看起来还是比较复杂的,不过Spring AI还提供更简易的方式:直接在call()
后面调用entity
,把对应的class类型传递进去即可,并且在提示词中**{format}**
占位符不需要再手动添加。
@GetMapping("/film/entity")
public Film queryFilmEntity(@RequestParam(value = "filmName", defaultValue = "无间道") String filmName) {Film resp = chatClient.prompt().user(u -> {u.text("""从豆瓣上查询影视作品{filmName}的信息 """).param("filmName", filmName);}).call().entity(Film.class);return resp;
}
输出:
{
“name”: “无间道”,
“releaseDate”: “2002-12-12”,
“directorName”: “刘伟强、麦兆辉”,
“desc”: “《无间道》是2002年上映的一部香港警匪片,讲述了警方和黑帮互相派遣卧底的故事,展现了复杂的身份认同和道德困境。”
}
三、集合类输出处理
3.1 列表输出:ListOutputConverter
@GetMapping("/list/converter/films")
public List<String> toListFilms(@RequestParam(value = "idol", defaultValue = "刘德华") String idol) {ListOutputConverter listOutputConverter = new ListOutputConverter(new DefaultConversionService());String content = chatClient.prompt().user(u -> {u.text("""帮我找五部{idol}主演的电影{format}""").params(Map.of("idol", idol, "format", listOutputConverter.getFormat()));}).call().content();List<String> list = listOutputConverter.convert(content);log.info("ListOutputConverter#toListFilms,size={},resp={}", list.size(), list);return list;
}
输出:
[
“无间道”,
“暗战”,
“天下无贼”,
“十面埋伏”,
“拆弹专家”
]
简易写法
@GetMapping("/list/converter/films/entity")
public List<String> toListFilmsEntity(@RequestParam(value = "idol", defaultValue = "刘德华") String idol) {List<String> list = chatClient.prompt().user(u -> {u.text("""帮我找五部{idol}主演的电影""").params(Map.of("idol", idol));}).call().entity(new ListOutputConverter(new DefaultConversionService()));log.info("ListOutputConverter#toListFilms,size={},resp={}", list.size(), list);return list;
}
3.2 Map输出:MapOutputConverter
@GetMapping("/map/converter/films")
public Map<String, Object> toMapFilms(@RequestParam(value = "style", defaultValue = "华语流行") String style) {MapOutputConverter mapOutputConverter = new MapOutputConverter();String content = chatClient.prompt().user(u -> {u.text("""帮我找五部{style}的电影,以电影名为分组键,值为电影信息,电影信息需要包含电影名称、上映时间、导演名、电影简介等内容{format}""").params(Map.of("style", style, "format", mapOutputConverter.getFormat()));}).call().content();Map<String, Object> resp = mapOutputConverter.convert(content);log.info("MapOutputConverter#toListFilms,size={},resp={}", resp.size(), resp);return resp;
}
输出:
{
“卧虎藏龙”: {
"电影名称": "卧虎藏龙","上映时间": "2000年","导演名": "李安","电影简介": "讲述一代大侠李慕白有退出江湖之意,托付红颜知己俞秀莲将自己的青冥剑带到京城,作为礼物送给贝勒爷收藏的故事。"
},
“让子弹飞”: {
"电影名称": "让子弹飞","上映时间": "2010年","导演名": "姜文","电影简介": "讲述了悍匪张牧之摇身一变化名清官“马邦德”上任鹅城县长,并与镇守鹅城的恶霸黄四郎展开一场激烈争斗的故事。"
},
“霸王别姬”: {
"电影名称": "霸王别姬","上映时间": "1993年","导演名": "陈凯歌","电影简介": "影片围绕两位京剧伶人半个世纪的悲欢离合,展现了对传统文化、人的生存状态及人性的思考与领悟。"
},
“大话西游之大圣娶亲”: {
"电影名称": "大话西游之大圣娶亲","上映时间": "1995年","导演名": "刘镇伟","电影简介": "讲述了至尊宝为了救白晶晶而穿越回到五百年前,遇见紫霞仙子之后发生一段感情并最终成长为孙悟空的故事。"
},
“无间道”: {
"电影名称": "无间道","上映时间": "2002年","导演名": "刘伟强、麦兆辉","电影简介": "讲述了两个身份混乱的男人分别为警方和黑社会的卧底,经过一场激烈的角斗,他们决心要寻回自己的故事。"
}
}
简易写法,要传递的entity是new ParameterizedTypeReference
类型, 和list还是有些细微的区别的
@GetMapping("/map/converter/films/entity")
public Map<String, Object> toMapFilmsEntity(@RequestParam(value = "style", defaultValue = "华语流行") String style) {Map<String, Object> resp = chatClient.prompt().user(u -> {u.text("""帮我找五部{style}的电影,以电影名为分组键,值为电影信息,电影信息需要包含电影名称、上映时间、导演名、电影简介等内容""").params(Map.of("style", style));}).call().entity(new ParameterizedTypeReference<Map<String, Object>>() {});log.info("MapOutputConverter#toListFilms,size={},resp={}", resp.size(), resp);return resp;
}
四、实体关系
Spring AI结构化输出转换器为开发者提供一种强大而灵活的方式来控制和解析LLM的输出。无论是需要简单的列表、复杂的映射还是完整的Java对象,这些转换器都能帮助我们轻松实现目标。通过合理使用这些工具,可以构建更可靠、更易维护的AI集成应用。
结语:让技术为生活服务
通过Spring AI的结构化输出转换器,成功:
- 将杂乱不稳定的模型输出转为整洁的Java对象
- 用几行代码打造私人影视推荐API
- 让周末选片时间从30分钟缩短到3秒
下次剧荒时,不妨试试这段代码:
@GetMapping("/weekend-recommendation")
public List<Film> getWeekendRecommendation() {return chatClient.prompt().user("推荐五部适合周末放松的优质电影").call().entity(new ParameterizedTypeReference<List<Film>>() {});
}
好的技术不仅应该解决工作问题,更应该让生活更美好——包括解决"看什么"这个重大生活课题!🎬
相关文章:
Spring AI 实战:第三章、Spring AI结构化输出之告别杂乱无章
引言:当程序员遇上剧荒 “周末看什么?” 这个看似简单的问题,往往能让我们在各大影视平台间反复横跳半小时,最后无奈选择重刷《老友记》。本期让我们用技术解决这个"世纪难题":让大模型成为你的私人影视推荐…...
ros2 humble 控制真实机械臂(以lerobot为例)
基础版 0.确保串口访问权限 sudo chmod 666 /dev/ttyARM0 # 确保串口访问权限 1.下载 lerobot 驱动功能包 git clone https://gitee.com/kong-yue1/lerobot_devices.git 2.编写控制节点(完整代码) 主要功能是与 Feetech 电机总线进行通信&#…...
REINFORCE蒙特卡罗策略梯度算法详解:python从零实现
🧠 向所有学习者致敬! “学习不是装满一桶水,而是点燃一把火。” —— 叶芝 我的博客主页: https://lizheng.blog.csdn.net 🌐 欢迎点击加入AI人工智能社区! 🚀 让我们一起努力,共创…...
模拟SIP终端向Freeswitch注册用户
1、简介 使用go语言编写一个程序,模拟SIP-T58终端在Freeswitch上注册用户 2、思路 以客户端向服务端Freeswitch发起REGISTER请求,告知服务器当前的联系地址构造SIP REGISTER请求 创建UDP连接,连接到Freeswitch的5060端口发送初始的REGISTER请…...
Elasticsearch 中的索引模板:如何使用可组合模板
作者:来自 Elastic Kofi Bartlett 探索可组合模板以及如何创建它们。 更多阅读: Elasticsearch:可组合的 Index templates - 7.8 版本之后 想获得 Elastic 认证吗?查看下一期 Elasticsearch Engineer 培训的时间! El…...
一篇文章看懂时间同步服务
Linux 系统时间与时区管理 一、时间与时钟类型 时钟类型说明管理工具系统时钟由 Linux 内核维护的软件时钟,基于时区配置显示时间timedatectl硬件时钟 (RTC)主板上的物理时钟,通常以 UTC 或本地时间存储,用于系统启动时初始化时间hwclock …...
Mysql常用语句汇总
Mysql语句分类 DDL: 数据定义语言,用来定义数据库对象(数据库、表、字段)DML: 数据操作语言,用来对数据库表中的数据进行增删改DQL: 数据查询语言,用来查询数据库中表的记录DCL: 数据控制语言,用来创建数据…...
迭代器的思想和实现细节
1. 迭代器的本质 迭代器是一种行为类似指针的对象,它可能是指针(如 std::vector 的迭代器),也可能是封装了指针的类(如 std::list 的迭代器)。如果是指针那天然就可以用下面的运算,如果是类&am…...
[Vue]编程式导航
在 Vue 中,编程式导航是通过 JavaScript 代码(而非 <router-link> 标签)动态控制路由跳转的核心方式。这个方法依赖于 Vue Router 提供的 API,能更灵活地处理复杂场景(如异步操作、条件跳转等)。 一、…...
使用Node.js搭建https服务器
一、引言 https是是http的安全版本,在http的基础上通过传输加密和身份认证保证了传输过程中的安全性。可以认为:https http tls/ssl。本文讲述使用Node.js搭建https服务器的方法。 二、编译OpenSSL 按照《Openssl在Linux下编译/交叉编译》࿰…...
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程
网络安全:sql注入练习靶场——sqli_labs安装保姆级教程 前言 sqli-labs靶场是一个开源的sql注入练习的综合靶场,包含大部分sql注入漏洞以及注入方式 网络安全学习者可以通过在sqli-labs靶场练习提升对sql注入的理解,以及学习各种绕过姿势。…...
rfsoc petalinux适配调试记录
1。安装虚拟机 2.设置共享文件夹 https://xinzhi.wenda.so.com/a/1668239544201149先设置文件夹路径 vmware 12 下安装 ubuntu 16.04 后,按往常的惯例安装 vmware-tools,安装时提示建议使用 open-vm-tools,于是放弃 vmware-tools 的安装&am…...
Windows下编译WebRTC源码
一、开发环境要求 准备一台64位的win10或win11(我用的是win11)电脑。最好是一台纯净的、没有安装过其它软件的Windows主机,避免已安装的软件和库对编译造成影响。 然后最好预留超过100G的硬盘空间。因为编译WebRTC时会产生大量的临时文件需…...
【vscode】.dart文件没有错误波浪线
解决方法: 新建一个文件夹,在vscode里打开这个文件夹 在这个文件夹里新建.dart文件后打开即可出现错误波浪线...
OpenharmonyOS+RK3568,【编译烧录】
文章目录 1. 摘要 ✨2. 代码下载 📩3. 编译 🖥️4. 修改&适配 ✂️4.1 编译框架基本概念4.2 vendor & device 目录4.3 内核编译4.3.1 如何修改、适配自己的开发板? 4.4 修改外设驱动 5. 烧录&验证 📋参考 1. 摘要 ✨ …...
从零开始理解 C++ 后端编程中的分布式系统
一、什么是“分布式”? 简单来说,分布式系统就是由“多个计算机(或服务器)”组成的一个大系统,它们通过网络协作完成某个任务,就像一个“团队合作”一样。 想象你开了一家餐馆,最初只有 一个厨房 和 一个服务员,所有订单都在这里处理。随着生意变好,你需要: 开分店…...
基于SpringBoot的篮球竞赛预约平台设计与实现
1.1 研究背景 科学技术日新月异的如今,计算机在生活各个领域都占有重要的作用,尤其在信息管理方面,在这样的大背景下,学习计算机知识不仅仅是为了掌握一种技能,更重要的是能够让它真正地使用到实践中去,以…...
具身系列——PPO算法实现CartPole游戏(强化学习)
完整代码参考: https://gitee.com/chencib/ailib/blob/master/rl/ppo_cartpole.py 执行结果: 部分训练得分: (sd) D:\Dev\traditional_nn\feiai\test\rl>python ppo_cartpole_v2_succeed.py Ep: 0 | Reward: 23.0 | Running: 2…...
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观
小程序与快应用:中国移动互联网的渐进式革命——卓伊凡的技术演进观 在知乎看到很多:“懂王”发布的要把内行笑疯了的评论,卓伊凡必须怼一下,真印证那句话,无知者无畏 一、Web与小程序的技术本质差异 1.1 浏览器渲染…...
Socket 编程 UDP
Socket 编程 UDP UDP 网络编程V1 版本 - echo serverV2 版本 - DictServerV3 版本 - 简单聊天室 补充参考内容地址转换函数关于 inet_ntoa UDP 网络编程 声明:下面代码的验证都是用Windows作为客户端的,如果你有两台云服务器可以直接粘贴我在Linux下的客…...
Lua 基础 API与 辅助库函数 中关于创建的方法用法
目录 基础 API 函数1. lua_len(L, index)2. lua_load(L, reader, data, chunkname, mode)3. lua_newstate(allocator, ud)4. lua_newtable(L)5. lua_newthread(L)6. lua_newuserdata(L, size)7. lua_next(L, index) 辅助库函数(luaL_*)8. luaL_len(L, in…...
YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测
这里写自定义目录标题 YOLOv11改进:利用RT-DETR主干网络PPHGNetV2助力轻量化目标检测1. 介绍2. 引言3. 技术背景3.1 YOLOv11概述3.2 RT-DETR与PPHGNetV23.3 相关工作 4. 应用使用场景5. 详细代码实现5.1 环境准备5.2 PPHGNetV2主干网络实现5.3 YOLOv11与PPHGNetV2集…...
centos7.0无法安装php8.2/8.3
在centos安装php8.2报错 configure: error: *** A compiler with support for C17 language features is required. 配置过程检测到你的系统编译器不支持 C17 语言特性,而 PHP 8.2 的编译需要编译器支持 C17 sudo yum update -y sudo yum install centos-releas…...
工业传动核心部件深度剖析:丝杆升降机与气缸的技术特性及选型指南
在工业自动化技术飞速发展的当下,丝杆升降机与气缸作为关键的直线传动部件,广泛应用于各类机械设备中。对于工程师而言,深入了解它们的技术特性、优缺点及适用场景,是实现高效、精准设备设计的重要前提。本文将从技术原理出发&…...
flask 获取各种请求数据:GET form-data x-www-form-urlencoded JSON headers 上传文件
在 Flask 里,能使用多种方法获取不同类型的请求数据,下面详细介绍常见请求数据的获取方式。 获取查询字符串参数(GET 请求) 查询字符串参数一般在 URL 里,以 ?key1value1&key2value2 这种形式存在。可通过 requ…...
c++_2011 NOIP 普及组 (1)
P1307 [NOIP 2011 普及组] 数字反转 P1307 [NOIP 2011 普及组] 数字反转 - 洛谷 # P1307 [NOIP 2011 普及组] 数字反转 ## 题目描述 给定一个整数 $N$,请将该数各个位上数字反转得到一个新数。新数也应满足整数的常见形式,即除非给定的原数为零&…...
信息泄露:网站敏感文件泄漏的隐形危机与防御之道
在网络安全领域,信息泄露常被称为“沉默的杀手”。攻击者无需复杂漏洞,仅通过网站无意暴露的敏感文件(如源码备份、配置文件、版本控制记录),即可获取数据库密码、API密钥甚至服务器权限。本文将深入剖析信息泄…...
C++笔记-多态(包含虚函数,纯虚函数和虚函数表等)
1.多态的概念 多态(polymorphism)的概念:通俗来说,就是多种形态。多态分为编译时多态(静态多态)和运行时多态(动态多态),这里我们重点讲运行时多态,编译时多态(静态多态)和运行时多态(动态多态)。编译时多态(静态多态)主要就是我们前面讲的函…...
2025年- H22-Lc130-206. 反转链表(链表)---java版
1.题目描述 2.思路 使用迭代法 (1)定义一个前指针 (2)然后定义两个变量 curr(head),curr.next。 (3)curr和curr.next交换位置(只要当前指针不为空,执行两两交换) 3.代码实现 /*** Definition for singly-…...
智能家居的OneNet云平台
一、声明 该项目只需要创建一个产品,然后这个产品里面包含几个设备,而不是直接创建几个产品 注意:传输数据使用到了不同的power,还有一定要手机先联网才能使用云平台 二、OneNet云平台创建 (1)Temperatur…...
二、shell脚本--变量与数据类型
1. 变量的定义与使用 定义变量:简单直接 在 Shell 里定义变量相当容易: 基本格式: variable_namevalue关键点 ❗:赋值号 的两边绝对不能有空格!这绝对是初学者最容易踩的坑之一 😨,务必留意!…...
GitHub Actions 和 GitLab CI/CD 流水线设计
以下是关于 GitHub Actions 和 GitLab CI/CD 流水线设计 的基本知识总结: 一、核心概念对比 维度GitHub ActionsGitLab CI/CD配置方式YAML 文件(.github/workflows/*.yml).gitlab-ci.yml执行环境GitHub 托管 Runner / 自托管GitLab 共享 Runner / 自托管市场生态Actions Mar…...
穿越数据森林与网络迷宫:树与图上动态规划实战指南
在 C 算法的浩瀚宇宙中,树与图就像是神秘的迷宫和茂密的森林,充满了未知与挑战。而动态规划则是我们探索其中的神奇罗盘,帮助我们找到最优路径。今天,就让我们一起深入这片神秘领域,揭开树与图上动态规划的神秘面纱&am…...
Java学习手册:Spring 生态其他组件介绍
一、微服务架构相关组件 Spring Cloud 服务注册与发现 : Eureka :由 Netflix 开源,包含 Eureka Server 和 Eureka Client 两部分。Eureka Server 作为服务注册表,接收服务实例的注册请求并管理其信息;Eureka Client 负…...
[android]MT6835 Android 移植brctl指令
说明 android默认brctl不支持showmacs选项,需要移植brctl-utils软件包 移除toybox中brctl编译 mssi/external/toybox/Android.bp 将 toybox_symlinks ["[","acpi","base64","basename","blockdev","br…...
安卓基础(悬浮窗分级菜单和弹窗)
initializeViews() 初始化 把全部的按钮都弄出来 // 主菜单按钮ImageButton mainButton floatingMenuView.findViewById(R.id.main_button);// 二级菜单按钮subButtons new ImageButton[3];subButtons[0] floatingMenuView.findViewById(R.id.sub_button_1);subButtons[1]…...
HTTP基础介绍+OSI七层参考模型+HTTP协议介绍
图片来源于网络 图片来源于网络 浏览器 Chrome:谷歌浏览器,推荐 Safari(WebKit):苹果浏览器,iOS,macOS Firefox:火狐浏览器,开源插件特别多(FireBug) IE:Wi…...
【项目实践】boost 搜索引擎
1. 项目展示 boost搜索引擎具体讲解视频 2. 项目背景 对于boost库,官方是没有提供搜索功能的,我们这个项目就是来为它添加一个站内搜索的功能。 3. 项目环境与技术栈 • 项目环境: ubuntu22.04、vscode • 技术栈: C/C、C11、S…...
接口隔离原则(ISP)
非常好,**接口隔离原则(ISP: Interface Segregation Principle)是 SOLID 五大原则中的第四个,它专门解决“一个接口太臃肿”**导致的麻烦。 我来从以下几个维度详细拆解: 🧠 什么是接口隔离原则࿱…...
Leetcode刷题记录29——矩阵置零
题源:https://leetcode.cn/problems/set-matrix-zeroes/description/?envTypestudy-plan-v2&envIdtop-100-liked 题目描述: 思路一: 💡 解题思路 本题中我们采用如下策略: 第一次遍历整个矩阵,记…...
复刻低成本机械臂 SO-ARM100 组装篇(打螺丝喽)
视频讲解: 复刻低成本机械臂 SO-ARM100 组装篇(打螺丝喽) 组装的视频有很多,参考大佬的《手把手复刻HuggingFace开源神作之Follower机械臂组装,资料已整理》_哔哩哔哩_bilibili,跟着视频做,大体…...
[更新完毕]2025东三省B题深圳杯B题数学建模挑战赛数模思路代码文章教学:LED显示屏颜色转换设计与校正
完整内容请看文章最下面的推广群 已经更新完整的文章代码 基于非线性映射与深度模型的多通道LED显示屏色彩校正 摘要 本研究聚焦于高动态色彩空间下LED显示屏的色彩映射与逐点校正问题,结合非线性回归理论与深度学习模型,构建了一套涵盖BT.2020映射、RG…...
Easy云盘总结篇-登录注册
**说在前面:该项目是跟着B站一位大佬写的,不分享源码,支持项目付费 ** 获取图形验证码 可以看到这里有2两种图形验证码,分为: type0:如上图下面那个,是完成操作后要进行注册的验证码 type1: 如…...
04 基于 STM32 的时钟展示程序
前言 我们经常会看到 各个场合下面有 基于数码管 的时钟程序 比如 在车站, 教室, 办公室 等等 各个场合都有 然后 这里就是做一个 简单的 时钟程序 展示程序 测试用例 每一秒钟更新时间, 然后 迭代更新 天, 时, 分 等等 然后 主流程 基于 天, 时分秒 渲染数码管 #incl…...
音视频开发技术总结报告
音视频开发技术总结报告 一、音视频开发基础 1、音频基础 声音原理 声波特性:频率、振幅、波长人耳听觉范围:20Hz-20kHz声音三要素:音调、音量、音色 数字音频基础 采样率:常见44.1kHz、48kHz、96kHz量化位数:8bit、…...
FastAPI系列13:API的安全防护
API的安全防护 1、HTTPS 强制什么是HTTPS强制如何在FastAPI中实现HTTPS强制 2、CORS跨域资源共享什么是CORS在 FastAPI 中开启 CORS 3、SQL注入防护什么是SQL注入如何在FastAPI中实现SQL注入防护 4、CSRF防护什么是CSRF防护如何在FastAPI中实现CSRF防护 在 FastAPI系列12&…...
每天一道面试题@第五天
1.包装类型的缓存机制了解么? 指部分包装类在创建对象时,会将一定范围内的对象缓存起来,当再次使用相同值创建对象时,优先从缓存中获取,而不是重新创建新对象。【提高性能】【节省内存】 列举几个常见的包装类缓存机…...
Python硬核革命:从微控制器到FPGA的深度开发指南
1. 重新定义硬件开发:Python的颠覆性突破 传统硬件开发长期被C/C++和Verilog/VHDL统治,但Python正通过两条路径改变这一格局: 1.1 微控制器领域的MicroPython革命 完整Python 3.4语法支持,运行在资源受限的MCU上(最低要求:64KB ROM,16KB RAM) 直接内存访问能力,突破…...
WebRTC 服务器之Janus概述和环境搭建
1 概述 Janus 是由 Meetecho 开发的通用 WebRTC 服务器,它为构建 WebRTC 应用程序提供了一个模块化框架。服务器目标:Janus WebRTC 网关被设计为轻量级、通用的 WebRTC 服务器,除了实现以下方法外,它本身不提供任何功能࿱…...
mcp+llm+rag
MCPRAG简介 前言一、MCP是什么?二、MCP工作原理(1. MCP Hosts(主机)(2.MCP Clients(客户端)(3. MCP Servers(服务端)(4. Local Data Sources(本地数据源&…...