Godot 4.x ECS插件GECS:数据驱动架构提升游戏性能与可维护性

news2026/5/13 9:08:43
1. 项目概述GECS为Godot 4.x注入ECS架构之力如果你正在用Godot开发游戏尤其是那种实体数量多、交互逻辑复杂的项目比如RTS、模拟经营或者一个满屏敌人的弹幕游戏你很可能已经感受到了传统面向对象OOP或纯节点Node架构的力不从心。实体管理混乱、性能瓶颈、代码耦合度高——这些问题在项目规模扩大后会变得尤为突出。今天要聊的GECS就是专门为解决这些问题而生的。它是一个为Godot 4.x量身打造的实体组件系统Entity-Component-System插件它的核心目标很明确通过数据与逻辑的彻底分离构建可伸缩、易维护的高性能游戏架构。简单来说GECS让你不再把“一个敌人”看作一个继承了所有功能的Enemy节点而是将其拆解一个代表身份的Entity实体一堆描述其属性的Component组件如Health、Position、Velocity以及一系列处理这些组件的System系统如MovementSystem、DamageSystem。这种范式转变带来的好处是巨大的数据驱动让缓存友好性能飙升组合优于继承让代码灵活易于复用关注点分离让逻辑清晰调试简单。GECS最聪明的地方在于它没有试图取代Godot强大的节点系统而是选择与其无缝集成。你依然可以使用熟悉的场景Scene、节点Node和编辑器同时享受ECS带来的架构优势。这意味着你可以用节点来管理渲染、物理碰撞和用户输入这些“表现层”的东西而用GECS来管理游戏核心的“逻辑层”状态和行为两者各司其职相得益彰。2. 核心架构与设计哲学解析2.1 为什么是ECS传统Godot开发模式的瓶颈在深入GECS之前我们得先搞清楚为什么需要它。传统的Godot开发我们习惯为每种游戏对象创建一个Node或Node2D/Node3D的子类。比如一个Player节点它可能有_physics_process处理移动有take_damage方法处理受伤属性如health、speed直接作为成员变量。当游戏里只有玩家和几个敌人时这很直观。但随着实体类型和数量增加问题接踵而至类爆炸Class Explosion想要一个会飞、会治疗、还会隐身的敌人你可能需要创建FlyingEnemy、HealingEnemy或者更糟多重继承Godot不支持导致你不得不复制代码或使用别扭的组合。紧耦合Tight Coupling移动逻辑、攻击逻辑、状态机逻辑全部塞在一个巨大的脚本里改一处而动全身测试和调试如同噩梦。缓存不友好Cache Unfriendly在_process里遍历所有Enemy节点来更新位置时CPU需要从内存各处跳跃着获取每个节点的不同数据位置、速度、生命值效率低下。系统逻辑分散一个“燃烧”效果可能需要遍历所有实体检查是否有Burnable组件并更新其状态这种跨实体的逻辑在OOP中很难优雅地集中处理。ECS架构正是针对这些痛点。Entity只是一个轻量的ID或容器它本身没有任何行为。Component是纯数据结构的“标签”或“属性包”比如Position {x, y}Health {value}。System是纯逻辑函数它遍历所有拥有特定组件组合的实体并对这些组件的数据进行操作例如MovementSystem遍历所有拥有Position和Velocity组件的实体更新其Position。2.2 GECS的设计亮点与Godot的共生而非取代很多ECS框架要求你完全抛弃原有的游戏引擎范式学习曲线陡峭。GECS则采取了更务实的“渐进式”路径实体即节点在GECS中Entity类继承自Node。这意味着你可以直接把一个Entity添加到场景树中它拥有所有节点的特性如process回调、分组、信号。你也可以将现有的任何Node“转换”为一个Entity为其附加组件。这种设计极大地降低了迁移成本。组件即资源GECS的Component类设计巧妙它通常以class_name脚本形式存在并且鼓励使用export变量。这使得组件的属性可以直接在Godot编辑器的检查器Inspector面板中可视化地编辑和配置实现了“数据驱动设计”的终极形态。你可以像调整材质参数一样在编辑器中调整实体的生命值、速度等属性。世界World与查询QueryGECS有一个全局的ECS.world单例你也可以创建多个世界用于隔离所有实体和系统都在其中注册。其核心引擎是查询系统。系统通过定义query()方法来声明它需要处理哪些实体例如“所有同时拥有Velocity和Position组件的实体”。GECS的查询引擎内部使用了高效的缓存和索引确保即使实体数量成千上万每次查询也近乎常数时间复杂度这是性能的关键。关系Relationship这是GECS超越基础ECS模型的一个强大特性。除了组件实体之间还可以建立关系。例如玩家实体可以有一个“持有武器”的关系指向一个武器实体。系统可以查询“所有持有某种武器的实体”这使得表达复杂的游戏逻辑如库存系统、队伍系统、空间父子关系变得异常清晰和高效。注意初次接触ECS时最大的思维转变是从“这个对象是什么Is-A”转向“这个对象拥有什么Has-A”。不要想着“这是一个敌人”而是想“这是一个实体它拥有生命值组件、移动组件和敌人标签组件”。系统只关心组件不关心实体具体代表什么。3. 从零开始GECS环境搭建与第一个实例3.1 插件安装的三种方式与实战选择GECS的安装非常灵活你可以根据团队协作习惯和项目阶段来选择。方式一Godot资产库安装最适合快速原型验证这是最无脑的方式。在Godot编辑器内点击顶部的“AssetLib”标签页在搜索框输入“GECS”找到插件后点击“Install”。安装完成后进入项目设置Project Settings - 插件Plugins找到GECS并启用它。这种方式适合个人项目或快速尝鲜但缺点是版本可能不是最新的且不便于版本控制。方式二手动复制最稳定可控前往GECS的GitHub发布页下载最新版本的Source code (zip)。解压后将其中的addons/gecs文件夹完整地复制到你Godot项目的addons/目录下如果没有就创建一个。同方式一在项目设置的插件页面启用GECS。 这种方式让你对插件文件有完全的控制权适合需要稳定版本、不希望依赖外部网络的项目。方式三Git子模块最适合团队协作与长期项目如果你的项目本身使用Git进行版本控制这是最佳实践。它能确保所有协作者使用完全相同的插件版本。# 在你的项目根目录下执行 git submodule add -b release-v6.8.1 https://github.com/csprance/gecs.git addons/gecs执行后GECS仓库会作为子模块链接到你的项目中。别忘了初始化并更新子模块git submodule update --init --recursive。之后同样需要在Godot编辑器中启用插件。这种方式将插件版本锁定在特定的提交或分支如示例中的release-v6.8.1避免了因插件更新意外破坏项目的情况。实操心得对于严肃的商业项目或团队项目我强烈推荐方式三Git子模块。它清晰地将第三方依赖与你的核心代码分离版本控制一目了然。启用插件后你会在编辑器顶部菜单栏看到“GECS”菜单里面包含调试查看器等工具这是检查插件是否成功加载的好方法。3.2 五分钟创建第一个ECS实体组件定义与实体组装理论说再多不如动手。我们来创建一个最简单的例子一个会在屏幕上移动的点。第一步定义组件纯数据在Godot中创建两个新的GDScript文件。# HealthComponent.gd class_name C_Health extends Component # 必须提供默认值否则编辑器会报错 export var max_health: int 100 export var current_health: int 100# VelocityComponent.gd class_name C_Velocity extends Component export var direction: Vector2 Vector2.RIGHT export var speed: float 100.0 # 可选提供一个带参数的构造函数方便代码创建 func _init(dir: Vector2 Vector2.RIGHT, spd: float 100.0) - void: direction dir speed spd关键点类名以C_开头是社区常见约定便于一眼区分组件和其他类。必须继承Component。export变量让数据可在编辑器调整且必须赋予默认值这是Godot GDScript 2.0的要求。第二步创建实体并附加组件你可以完全用代码创建也可以在场景中创建。代码方式extends Node2D func _ready(): # 1. 创建实体也是一个节点 var player_entity Entity.new() player_entity.name PlayerEntity add_child(player_entity) # 添加到场景树 # 2. 创建并添加组件 var health_comp C_Health.new() health_comp.max_health 150 # 可以覆盖默认值 player_entity.add_component(health_comp) var velocity_comp C_Velocity.new(Vector2(1, 0.5).normalized(), 80.0) player_entity.add_component(velocity_comp) # 3. 将实体注册到ECS世界重要 ECS.world.add_entity(player_entity)场景编辑器方式在场景中创建一个Node2D作为根。为其添加一个脚本在_ready()中调用convert_to_entity()或者直接添加一个Entity节点如果插件提供了该节点类型。选中该实体节点在检查器Inspector面板你会看到一个“Components”分组。点击“Add Component”可以搜索并添加你刚创建的C_Health和C_Velocity组件并直接在面板上修改其属性值。这种方式对设计师和非程序员朋友极其友好。第三步定义系统纯逻辑创建一个处理移动的系统。# MovementSystem.gd class_name MovementSystem extends System # 定义查询本系统只处理同时拥有Transform2DGodot内置和C_Velocity组件的实体 func query() - QueryBuilder: return q.with_all([Transform2D, C_Velocity]) # 处理函数对查询到的每个实体执行逻辑 func process(entities: Array[Entity], components: Array, delta: float) - void: for entity in entities: # 获取该实体的Velocity组件 var vel_comp: C_Velocity entity.get_component(C_Velocity) # 获取该实体的Transform2D组件来自其父Node2D var transform_comp: Transform2D entity.get_component(Transform2D) # 计算位移 var movement vel_comp.direction * vel_comp.speed * delta # 更新位置这里直接操作组件的属性 transform_comp.origin movement第四步注册系统并驱动执行在你的主场景如一个Node2D的脚本中extends Node2D func _ready(): # 注册系统 ECS.world.add_system(MovementSystem.new()) # 可以注册更多系统... # ECS.world.add_system(CollisionSystem.new()) # ECS.world.add_system(DamageSystem.new()) func _process(delta: float) - void: # 每一帧驱动ECS世界更新它会按顺序执行所有已注册系统的process方法 ECS.process(delta)运行游戏你会发现拥有C_Velocity组件的实体开始移动了整个过程中数据位置、速度和逻辑移动计算是清晰分离的。4. 核心机制深度剖析查询、关系与观察者4.1 强大的查询系统如何精准定位实体查询Query是ECS架构的“心脏”。GECS的查询构建器QueryBuilder提供了极其灵活的方式来筛选实体。q是一个全局的查询构建器助手。基础查询q.with_all([C_A, C_B]): 查找同时拥有组件A和B的实体。最常用。q.with_any([C_A, C_B]): 查找拥有至少一个组件A或B的实体。q.with_none([C_A]): 查找不拥有组件A的实体。组合查询链式调用func query() - QueryBuilder: return ( q.with_all([C_Health, C_Transform]) # 必须有生命和变换 .with_any([C_Player, C_Enemy]) # 并且是玩家或敌人 .with_none([C_Dead]) # 并且不是死亡状态 )这个系统将处理所有活着的、有位置信息的玩家或敌人实体。基于组件属性的查询高级这是GECS非常强大的功能。你不仅可以按组件类型筛选还可以按组件属性的值来筛选。func query() - QueryBuilder: return ( q.with_all([C_Health]) .where(C_Health, current_health, , 50) # 生命值低于50的实体 )where方法支持多种比较操作符,!,,,,甚至可以结合and/or进行复杂条件组合。这使得实现诸如“寻找附近生命值最低的友军”这样的逻辑变得非常简单高效因为过滤是在高度优化的查询引擎内部完成的而不是在GDScript的循环里。性能提示GECS会缓存查询结果。如果一个系统的查询条件没有变化且相关的组件类型没有实体被添加或删除那么process调用中获得的entities数组会是缓存的结果避免了每帧重复进行昂贵的匹配计算。这意味着定义好查询后你可以放心地在_process中调用ECS.process。4.2 构建实体网络关系组件的妙用组件描述实体的内在属性而关系Relationship描述实体之间的外在联系。在GECS中关系本质上也是一种特殊的组件它连接两个实体。典型应用场景库存系统玩家实体has_a武器实体。空间层级一个飞船实体parent_of多个炮台实体。队伍系统单位实体ally_of另一个单位实体。目标锁定导弹实体targeting敌机实体。如何使用首先定义一个关系组件它通常继承自Relationship或是一个简单的标记组件。# 定义一个“持有”关系 class_name R_Holding extends Relationship # 关系组件本身也可以有数据比如持握位置偏移 export var grip_offset: Vector3 Vector3.ZERO然后在代码中建立关系var player ECS.world.create_entity() # 快捷创建方法 var sword ECS.world.create_entity() # 玩家持有剑。关系是有方向的从玩家指向剑。 player.add_relationship(R_Holding.new(), sword)在系统中查询关系func query() - QueryBuilder: # 查询所有持有R_Holding关系的实体 return q.with_all([R_Holding]) func process(entities: Array[Entity], delta: float) - void: for holder in entities: var holding_rel: R_Holding holder.get_component(R_Holding) var sword_entity: Entity holding_rel.target_entity # 现在你可以更新剑的位置使其跟随玩家 if sword_entity and sword_entity.has_component(Transform3D): var sword_transform sword_entity.get_component(Transform3D) var holder_transform holder.get_component(Transform3D) sword_transform.origin holder_transform.origin holding_rel.grip_offset关系查询同样强大你可以查询“所有持有某种特定实体比如ID为123的剑的玩家”或者“所有被任何实体持有的物品”。4.3 响应式编程观察者模式与事件处理在游戏中我们经常需要响应状态变化生命值降到零触发死亡拾取物品触发效果碰撞发生触发伤害。在传统代码中这通常通过信号Signal或直接函数调用来实现容易导致复杂的依赖网。GECS提供了观察者Observer作为一种优雅的响应式解决方案。观察者是一种特殊的系统它不是在每帧主动运行而是在特定事件发生时被触发。主要事件类型OnComponentAdded: 当某个组件被添加到实体时。OnComponentRemoved: 当某个组件从实体移除时。OnComponentChanged: 当某个组件的属性值发生变化时需要组件实现特定的接口来通知变化。示例死亡观察者class_name DeathObserver extends Observer # 观察者也需要定义查询来限定它关心哪些实体/组件的变化 func query() - QueryBuilder: return q.with_all([C_Health]) # 当C_Health组件被添加到一个新实体时虽然不常见或者更常见的是我们监听变化 # 这里我们假设C_Health组件有一个_on_current_health_changed的回调需自己实现信号或setter # 更实用的模式是另一个DamageSystem会修改C_Health.current_health并在值0时添加一个C_Dead标记组件。 # 然后我们可以用一个系统来处理所有拥有C_Dead组件的实体。 # 但为了演示观察者假设我们监听组件添加 func on_component_added(entity: Entity, component: Component) - void: if component is C_Health: print(Entity , entity, now has health!) # 更强大的用法监听C_Health组件的current_health属性变化需要配置 # 这通常需要你在C_Health组件中使用setter并发出通知。实际上更经典的ECS模式是用组件状态变化来驱动系统而非严格的事件监听。例如DamageSystem遍历所有受到攻击的实体减少其C_Health.current_health。在DamageSystem内部如果发现current_health 0则给该实体添加一个C_Dead标签组件。另一个DeathCleanupSystem的查询是q.with_all([C_Dead])。它会处理所有死亡实体播放死亡动画、掉落物品、从世界移除等。处理完毕后DeathCleanupSystem会移除C_Dead组件或直接销毁实体。这种“添加/移除组件作为事件”的模式是ECS中非常典型和高效的状态管理方式观察者模式可以在此基础上提供更细粒度的响应。5. 性能优化与调试实战指南5.1 让游戏飞起来GECS性能优化核心策略ECS架构本身就是为了性能而生但使用不当仍会拖后腿。以下是针对GECS的优化要点1. 组件设计原则小而纯保持组件轻量组件应只包含数据尽可能使用基础类型int,float,Vector2。避免在组件中存储复杂的对象引用或数组。如果需要存储一个EntityID或资源ID在系统中通过ID去查找。避免在组件中嵌入逻辑组件的_init或_ready里不要做复杂计算。逻辑属于系统。使用标记组件Tag Component如果一个组件只用于标记状态如C_Dead,C_PlayerControlled不需要任何数据字段可以创建一个空类。这比用布尔值组件更高效因为查询引擎处理类型过滤比属性过滤更快。2. 系统设计与查询优化合并系统如果两个系统总是遍历同一组实体且逻辑简单考虑合并它们以减少遍历开销。但平衡可读性与性能不要过度合并。善用查询缓存如前所述GECS自动缓存查询。确保系统的query()方法返回的QueryBuilder条件是稳定的。不要在query()内部动态生成条件除非必要这会导致缓存失效。减少每帧的查询次数如果某个数据在多个系统中都需要考虑在一个系统中计算并存储到一个共享的“单例组件”一个附着在特定实体上的组件供其他系统查询获取而不是每个系统都去计算一遍。分帧处理对于非实时要求的系统如AI决策、寻路更新不要每帧都运行。可以设置一个计时器每N帧运行一次或者根据距离玩家的远近设置不同的更新频率。3. 内存与实例化优化对象池Object Pooling对于频繁创建和销毁的实体如子弹、特效不要直接new Entity()和queue_free()。使用对象池预先创建一批实体使用时激活并重置组件用完后回收到池中。GECS本身不提供池但你可以很容易地基于Entity实现一个。批量操作GECS的process方法传入的是当前帧所有匹配的实体数组。尽量在系统内部使用简单的循环避免在循环内进行复杂的查询或创建新实体。4. 与Godot渲染/物理的交互渲染组件可以创建一个C_Sprite2D组件其内部持有一个Sprite2D节点的引用。RenderingSystem遍历所有有C_Sprite2D和C_Transform的实体更新Sprite2D节点的位置。这样渲染逻辑也纳入了ECS管理。物理组件类似地可以创建C_RigidBody2D组件PhysicsSystem负责同步ECS中的C_Velocity,C_Transform与Godot物理引擎RigidBody2D的状态。注意物理引擎通常也在_physics_process中更新你需要协调好ECS的process和Godot的_physics_process的调用顺序。5.2 调试利器GECS调试查看器与常见问题排查再好的架构也离不开调试。GECS内置了一个强大的实时调试查看器Debug Viewer。启用与使用启用插件后在编辑器顶部菜单栏点击GECS - Open Debug Viewer。你会看到一个独立的窗口通常包含以下面板实体列表Entities显示世界中所有实体的ID和名称。组件列表Components显示所有已注册的组件类型。系统列表Systems显示所有已注册的系统及其当前状态是否激活。实体详情点击一个实体可以查看它身上挂载的所有组件及其当前属性值。你甚至可以在运行时直接修改这些属性这对调试平衡性数值伤害、速度极其有用。性能监控可能会显示每个系统的执行时间帮助你定位性能热点。常见问题与排查技巧实体没有移动/系统没执行检查是否在_ready中调用了ECS.world.add_entity(entity)实体必须注册到世界。检查是否在_ready中调用了ECS.world.add_system(system)系统必须被注册。检查主循环如_process中是否调用了ECS.process(delta)这是驱动所有系统运行的引擎。检查系统的query()方法是否正确用Debug Viewer查看该系统的匹配实体数是否为0。检查组件是否被正确添加在Debug Viewer中选中实体查看其组件列表。查询性能突然下降可能原因某个系统每帧都在动态修改其query()条件导致缓存频繁失效。尽量使用静态查询。可能原因实体数量剧增。考虑是否需要使用空间分割如网格、四叉树来优化某些查询如“寻找附近的敌人”GECS的基础查询是按组件类型过滤空间查询需要额外实现或结合Godot的Area2D。编辑器里修改组件属性不生效注意在编辑器中为实体节点添加组件并设置属性这些属性值是在_ready()之前就设置好的。如果你的系统在_ready中创建实体并添加组件会覆盖编辑器设置。确保逻辑顺序正确或者考虑在_init或_enter_tree阶段处理组件初始化。多场景切换时实体泄露清理当切换场景时旧场景中的实体可能还留在ECS.world中。你需要在场景卸载前如_tree_exiting信号中手动遍历并调用ECS.world.remove_entity(entity)或者更粗暴地调用ECS.world.clear()清空整个世界。更好的模式是为每个游戏关卡创建一个独立的ECS.World实例而非使用全局单例。与Godot节点通信困难模式记住Entity就是Node。你可以用entity.get_node()来获取其子节点也可以用entity.emit_signal()发射Godot信号。对于需要与UI如血条交互的情况可以创建一个C_HealthChanged事件组件由一个专门的UISystem来消费这个事件并更新UI。

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

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

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…