C# AR应用性能优化三大硬核策略

news2026/5/24 4:44:41
1. 这不是“加个特效”就能解决的问题AR应用卡顿背后的真实战场C# AR应用优化实战——这七个字我盯着看了三分钟。不是因为难懂而是因为太熟悉了。过去三年我带过7个AR项目从工业设备远程巡检到博物馆文物交互导览从UnityAR Foundation到自研轻量引擎OpenXR几乎踩遍了所有能踩的坑。而每次客户说“体验不够流畅”90%的情况根本不是渲染管线没调好也不是手机性能差而是我们把AR当成了“带摄像头的3D游戏”忽略了它最本质的约束实时性、空间一致性、传感器耦合性。你可能试过把Draw Call压到50以下帧率还是上不去可能把模型面数砍掉80%用户依然抱怨“转头就卡顿”甚至开了GPU Instancing结果在中端安卓机上内存直接爆掉。这不是玄学是C#层面对AR运行时状态的误判——比如你有没有在Update里每帧都调用WorldAnchorManager.GetAnchors()有没有在OnTrackingChanged回调里偷偷做了同步IO有没有把ARSessionOrigin的重定位逻辑和UI动画绑在同一个协程里这些操作在普通Unity项目里可能只是小毛刺在AR里就是致命延迟源。本文讲的“3大核心策略”不是泛泛而谈的“降低分辨率”“减少粒子”而是我在产线实测中验证过的、可量化、可复现、可嵌入CI/CD流程的硬核方案帧级资源调度策略、空间锚点生命周期治理、传感器-渲染双线程解耦架构。它们共同作用让某款工业AR巡检App在华为Mate 40 Pro麒麟9000上平均帧率从28.6fps提升至40.3fps卡顿率下降42.7%最关键的是——用户主观“眩晕感”评分下降58%。适合正在做Unity AR项目的开发者、技术负责人以及被“体验差”反复背锅的客户端工程师。如果你的AR应用还在靠“换台好手机”来解决问题那这篇就是为你写的。2. 帧级资源调度为什么“按需加载”在AR里会失效2.1 传统AssetBundle加载逻辑的三大反AR特性在常规Unity项目里“按需加载”是金科玉律用户点开某个模块再加载对应资源内存友好启动快。但AR场景下这套逻辑会引发灾难性后果。我拿一个真实案例说明某汽车维修AR指导系统需要根据用户扫描的发动机舱区域动态加载对应部件的3D模型与维修步骤动画。开发团队沿用标准AssetBundle流程——检测到ARPlane出现触发Bundle.LoadFromFileAsync()等Load完成再Instantiate。结果呢用户扫到油底壳等待2.3秒才看到模型浮现转头扫到火花塞又等1.8秒。这不是加载慢是时机错配。AR的物理世界是连续演进的而你的资源加载是离散触发的。更糟的是AssetBundle.LoadFromFileAsync()在Android上实际走的是主线程文件I/O即使标为Async在低端机上极易阻塞主线程导致CameraTexture更新延迟画面撕裂。我们做过对比测试同一台Redmi Note 10骁龙732G纯内存加载10MB模型耗时112ms而从AssetBundle加载同等模型平均耗时487ms其中310ms是I/O阻塞时间。这不是Unity的锅是AR对“实时响应”的刚性要求撞上了传统资源管理的异步假象。提示Unity官方文档里写的“LoadFromFileAsync is non-blocking”仅指托管堆分配不阻塞但底层文件读取仍可能抢占主线程磁盘带宽尤其在eMMC存储的中低端安卓设备上。2.2 帧级预加载用“空间预测”替代“事件触发”解决方案不是放弃按需而是把“按需”升级为“预需”。核心思想利用AR空间理解能力提前一帧预测用户即将关注的区域并在此刻预加载资源。我们不再监听ARPlaneAdded而是监听ARSession.updatedAnchors从中提取所有已跟踪平面的边界框Bounds结合当前Camera.main.transform.forward方向计算出“视线锥体Frustum内且距离3m的平面集合”。然后对这些平面打上业务标签如“EngineBay”“WheelWell”并查表映射到待加载的AssetBundle ID。关键在于执行时机——我们把这个预测逻辑放在LateUpdate末尾而加载动作放在下一帧的Start()里。为什么是Start()因为Start()保证在所有Update之后、所有渲染之前执行此时Camera pose已稳定Frustum计算最准且不会干扰本帧渲染。代码骨架如下// 在ARSessionManager单例中 private Liststring _pendingBundles new Liststring(); private HashSetstring _loadedBundles new HashSetstring(); void LateUpdate() { if (!ARSession.state.Equals(ARSessionState.SessionTracking)) return; var frustumPlanes GeometryUtility.CalculateFrustumPlanes(Camera.main); var candidatePlanes new ListARPlane(); foreach (var anchor in ARSession.updatedAnchors) { if (anchor is ARPlane plane plane.trackingState TrackingState.Tracking Vector3.Distance(plane.center, Camera.main.transform.position) 3f) { // 粗略判断是否在视锥内避免矩阵运算开销 if (GeometryUtility.TestPlanesAABB(frustumPlanes, plane.bounds)) { candidatePlanes.Add(plane); } } } // 根据平面中心点聚类合并相近平面防重复加载 var clusteredTags ClusterPlanesByTag(candidatePlanes); foreach (var tag in clusteredTags) { if (!_loadedBundles.Contains(tag) !_pendingBundles.Contains(tag)) { _pendingBundles.Add(tag); } } } void Start() { // 此处执行真正的加载确保在渲染前完成 foreach (var tag in _pendingBundles.ToList()) { StartCoroutine(LoadBundleAsync(tag)); } _pendingBundles.Clear(); }这个设计把资源加载从“被动响应”变为“主动预测”实测将首帧模型出现延迟从平均2.1秒压缩至0.35秒以内。更重要的是它规避了主线程I/O阻塞——因为LoadFromFileAsync()调用发生在Start()而Unity保证Start()在所有Update之后、所有渲染之前此时主线程相对空闲。2.3 内存热区管理让GPU显存也学会“呼吸”光解决加载延迟还不够。AR应用常需同时驻留多个高精度模型如整台发动机各子部件拆解图全加载进内存必然OOM。传统做法是“用完即卸”但AR里“用完”很难定义——用户可能扫完油底壳转头看轮胎3秒后又扫回油底壳。频繁加载卸载造成GPU显存抖动表现为模型闪烁、材质丢失。我们的方案是引入内存热区Hot Zone机制以Camera位置为球心半径2.5m内为热区1.5m内为超热区。热区内资源保持常驻超热区内资源优先使用GPU Instancing渲染热区外资源进入“休眠态”——即卸载Mesh数据但保留Material和Texture引用且不Destroy GameObject只SetActive(false)。这样下次进入热区时只需Rebuild MeshFilter耗时仅12~18msvs 全量Instantiate的80~120ms。我们用一个轻量级管理器实现public class ARResourceHotZone : MonoBehaviour { public float hotRadius 2.5f; public float superHotRadius 1.5f; private DictionaryGameObject, HotState _resourceStates new DictionaryGameObject, HotState(); void Update() { var camPos Camera.main.transform.position; foreach (var kvp in _resourceStates) { float dist Vector3.Distance(camPos, kvp.Key.transform.position); HotState newState dist superHotRadius ? HotState.SuperHot : dist hotRadius ? HotState.Hot : HotState.Cold; if (newState ! kvp.Value) { SwitchState(kvp.Key, kvp.Value, newState); _resourceStates[kvp.Key] newState; } } } void SwitchState(GameObject go, HotState from, HotState to) { switch (to) { case HotState.Cold: // 卸载Mesh保留材质纹理 var mf go.GetComponentMeshFilter(); if (mf ! null) { mf.mesh null; // 释放GPU显存 go.SetActive(false); } break; case HotState.Hot: go.SetActive(true); break; case HotState.SuperHot: go.SetActive(true); // 启用Instancing var mr go.GetComponentMeshRenderer(); if (mr ! null) mr.enabled true; break; } } }这套机制让某款AR维修App在小米12骁龙8 Gen1上GPU显存占用峰值得到平滑从剧烈波动的180~320MB稳定在210±15MB卡顿帧率下降37%。它本质上是把内存管理从“二值开关”升级为“多级缓存”贴合AR空间的连续性本质。3. 空间锚点生命周期治理别让“世界坐标”变成内存黑洞3.1 锚点泄漏的隐蔽性你以为的“自动回收”其实是幻觉AR应用里WorldAnchor或ARAnchor是连接虚拟与现实的基石。但它的生命周期管理是绝大多数C# AR项目最大的隐形内存杀手。很多人以为“只要ARSession重置锚点就自动销毁”这是严重误解。在AR Foundation中Anchor对象本身是C#托管对象其底层Native Anchor如ARKit的ARAnchor或ARCore的ArAnchor由平台SDK持有强引用。当你调用anchorManager.RemoveAnchor(anchor)时只是解除了C#层引用Native Anchor仍驻留在平台内存中直到ARSession完全销毁或平台主动GC——这个过程可能长达数分钟。我们曾用Unity Profiler抓取一个简单AR涂鸦App用户每画一笔创建一个Anchor10分钟后未清理Native内存增长127MB而Managed Heap仅显示增长8MB。这就是典型的“Native内存泄漏”Profiler里看不到但手机会发烫、后台被杀。更隐蔽的是某些AR SDK如旧版Vuforia在Anchor丢失追踪后不会自动释放Native资源必须手动调用Destroy()否则Anchor对象永远“活着”。注意Unity 2021.3版本中ARFoundation已改进Anchor生命周期但仅限于新API如ARAnchorManager老项目若混用ARSession.nativeSession.GetAnchors()等底层调用仍存在泄漏风险。3.2 锚点健康度评估用“空间置信度”替代“存活时间”传统锚点管理依赖计时器创建后30秒无更新就销毁。这在AR里极不靠谱——用户可能正专注观察一个静止物体锚点完美跟踪但“无更新”被误判为失效。我们的方案是引入空间置信度Spatial Confidence指标综合三个维度动态评估锚点健康度跟踪稳定性连续5帧内Anchor.pose.position变化标准差 0.003m约3mm环境光照适应性ARCameraManager.frame.lightEstimation.averageBrightness变化率 5%/s防光照突变误判几何一致性Anchor所在平面若关联ARPlane的面积变化率 10%/s且法线角度偏移 5°只有三项全部达标才视为“高置信度锚点”否则降级为“观察中”状态。我们用一个独立协程每秒评估一次private IEnumerator EvaluateAnchorHealth(ARAnchor anchor) { while (true) { yield return new WaitForSeconds(1f); if (!anchor.isValid || anchor.trackingState ! TrackingState.Tracking) continue; bool isStable IsPositionStable(anchor); bool isLightAdapted IsLightAdapted(); bool isGeometryConsistent IsGeometryConsistent(anchor); float confidence (isStable ? 0.4f : 0f) (isLightAdapted ? 0.3f : 0f) (isGeometryConsistent ? 0.3f : 0f); if (confidence 0.6f) { // 进入观察期连续2次低于阈值才触发清理 anchorHealth[anchor] (anchorHealth.GetValueOrDefault(anchor, 0) 1); if (anchorHealth[anchor] 2) { TryRemoveAnchor(anchor); } } else { anchorHealth[anchor] 0; // 重置计数 } } }这套机制让锚点误删率从23%降至1.7%同时Native内存泄漏率归零。它把锚点管理从“时间驱动”升级为“空间驱动”真正贴合AR的本质——空间感知。3.3 锚点批量回收协议避免GC风暴的“分时手术”即使精准识别了失效锚点也不能一股脑全删。ARFoundation中RemoveAnchor()内部会触发Native层资源释放若同时删除上百个锚点会造成短暂的CPU尖峰Native GC和GPU等待表现为1~2秒的全局卡顿。我们的解法是分时批量回收Time-Sliced Batch Removal将待删锚点列表按哈希分片每帧只处理一片。例如120个待删锚点分成12片每帧删10个持续12帧约200ms。代码实现极其简单private ListARAnchor _anchorsToDispose new ListARAnchor(); private int _disposeIndex 0; private const int DISPOSE_PER_FRAME 10; void Update() { if (_anchorsToDispose.Count 0 _disposeIndex _anchorsToDispose.Count) { int end Mathf.Min(_disposeIndex DISPOSE_PER_FRAME, _anchorsToDispose.Count); for (int i _disposeIndex; i end; i) { if (_anchorsToDispose[i].isValid) { anchorManager.RemoveAnchor(_anchorsToDispose[i]); } } _disposeIndex end; } }这个看似“偷懒”的设计实测将锚点清理引发的卡顿帧从平均4.2帧降至0.3帧。它本质上是把一次大手术拆成12次微创让系统始终有余力处理渲染和传感器数据。4. 传感器-渲染双线程解耦让陀螺仪数据不再“堵”在主线程4.1 主线程瓶颈的真相不是CPU算力是传感器数据流很多开发者优化AR性能时第一反应是“Profile CPU找耗时函数”。但我们在某款AR导航App中发现CPU Usage常年低于35%GPU Usage却高达92%而帧率只有22fps。深入分析Frame Debugger问题出在ARCameraManager.frame.timestamp——这个属性每次访问都会触发底层传感器数据同步而该同步操作是阻塞式的。在Unity 2020.3中ARFoundation默认每帧调用一次frame.get_timestamp()来校准渲染时间戳这在高端机上耗时0.8ms但在中端机如三星A52上飙升至4.2ms。更糟的是如果业务代码里还有类似ARSession.state、ARCameraManager.camera.transform.position的频繁访问这些调用会排队等待传感器锁形成“数据流堵塞”。我们用Unity的Deep Profile模式抓取发现主线程37%的时间花在ARCoreSession::GetLatestFrame()的等待上。这不是算法问题是架构问题——把实时性要求最高的传感器数据和实时性要求次高的渲染逻辑强行绑在同一根线上。4.2 双线程管道用RingBuffer构建传感器数据高速公路解决方案是彻底解耦传感器采集与渲染分离用无锁环形缓冲区Lock-Free RingBuffer传递数据。我们创建一个独立的Native PluginC编写在Android端直接调用ARCore的ArSession_getAllAnchors()和ArFrame_getCameraPose()以60Hz频率采集原始位姿数据写入共享内存RingBuffer。C#层则用一个低优先级线程ThreadPriority.BelowNormal轮询该Buffer解析出Camera Pose、Light Estimation等结构体存入线程安全的ConcurrentQueue。主渲染线程Update只从此Queue中取最新一帧数据绝不触碰任何ARFoundation的托管属性。整个管道如下ARCore Native SDK → C Plugin60Hz采集 → Shared Memory RingBuffer → C# SensorPoller Thread轮询 → ConcurrentQueueARSensorData → Main ThreadUpdate中消费关键点在于RingBuffer的无锁设计。我们采用经典的Single-Producer-Single-ConsumerSPSC模式用原子操作Interlocked.CompareExchange管理读写指针避免Mutex开销。C侧代码核心片段// ringbuffer.h struct SPSCRingBuffer { std::atomicuint32_t writeIndex{0}; std::atomicuint32_t readIndex{0}; ARSensorData* buffer; uint32_t capacity; bool tryWrite(const ARSensorData data) { uint32_t w writeIndex.load(std::memory_order_acquire); uint32_t r readIndex.load(std::memory_order_acquire); if ((w 1) % capacity r) return false; // full buffer[w] data; writeIndex.store((w 1) % capacity, std::memory_order_release); return true; } };C#侧通过Marshal.PtrToStructure高效读取每帧耗时稳定在0.03ms以内vs 原生调用的4.2ms。这个架构让主线程彻底摆脱传感器依赖CPU Usage降至18%GPU Usage因渲染更稳定反而提升至95%但帧率跃升至38fps——因为GPU不再被主线程“饿着”。4.3 渲染时间戳校准用插值对抗传感器延迟解耦后带来新问题传感器数据有~16ms延迟60Hz采集周期而渲染需要精确时间戳。若直接用RingBuffer里的时间戳会导致虚拟物体“滞后”于真实世界。我们的校准方案是双时间戳插值Dual-Timestamp Interpolation在C Plugin中每次写入RingBuffer时同时记录两个时间戳nativeTimestampARCore返回的原始时间戳纳秒级但有延迟wallClockTimestamp调用clock_gettime(CLOCK_MONOTONIC)获取的系统单调时钟毫秒级无延迟C#层消费时用当前Time.unscaledTime与wallClockTimestamp计算延迟Δt再对nativeTimestamp做线性插值得到校准后的时间戳public struct ARSensorData { public long nativeTimestamp; // ARCore原始时间戳 public long wallClockTimestamp; // 系统单调时钟 public Pose cameraPose; // ... other fields } // 在SensorPoller线程中 private ARSensorData? _latestData; private float _calibrationOffset 0f; void ConsumeBuffer() { if (_queue.TryDequeue(out var data)) { _latestData data; // 计算校准偏移假设传感器延迟恒定 float delayMs (Time.unscaledTime * 1000f) - (data.wallClockTimestamp / 1_000_000f); _calibrationOffset Mathf.Lerp(_calibrationOffset, delayMs, 0.1f); } } // 在Main Thread Update中 void Update() { if (_latestData.HasValue) { float calibratedTime Time.unscaledTime - (_calibrationOffset / 1000f); // 使用calibratedTime驱动动画、插值等 AnimateObject(_latestData.Value.cameraPose, calibratedTime); } }这个插值模型把视觉延迟从平均21ms压缩至8.3ms用户主观“粘滞感”消失。它证明高性能AR不是堆硬件而是用软件工程思维把每个环节的延迟都当作可优化的变量。5. 效果验证与产线落地40%提升不是营销话术5.1 量化对比三组对照实验的设计逻辑所谓“用户体验提升40%”绝非拍脑袋的营销话术而是基于三组严格控制的对照实验。我们选定了三款典型AR应用作为测试载体A工业维修指导、B文旅导览、C教育解剖。每组实验均在相同硬件华为Mate 40 Pro、相同环境室内恒光实验室、相同测试脚本标准化用户操作路径下进行。关键指标不是单一帧率而是复合体验指数CEI由四个维度加权构成视觉流畅度40%权重Smoothness Score (1 - 0.01 × 长时间卡顿帧占比) × (1 - 0.005 × 短时抖动帧占比)空间一致性30%权重Alignment Score 1 - (平均虚拟物体漂移像素 / 屏幕宽度)交互响应度20%权重Responsiveness Score 1 - (平均操作到反馈延迟 / 200ms)系统稳定性10%权重Stability Score 1 - (崩溃/热重启次数 / 总测试时长)传统方案Baseline指未应用本文三策略的原始版本智能优化版Optimized指完整集成帧级调度、锚点治理、双线程解耦的版本。结果如下表应用类型Baseline CEIOptimized CEI提升幅度主要贡献策略工业维修A0.6230.87941.1%帧级调度18.2%、双线程解耦15.3%、锚点治理7.6%文旅导览B0.5870.83241.7%双线程解耦22.1%、帧级调度12.4%、锚点治理7.2%教育解剖C0.6410.89239.2%锚点治理19.8%、帧级调度13.5%、双线程解耦5.9%提示CEI是归一化指标0.879表示整体体验达到理论最优值的87.9%并非百分比数值本身。5.2 产线集成如何把策略变成CI/CD里的一个开关再好的策略若不能融入开发流程就是纸上谈兵。我们已将这三大策略封装为Unity Package支持一键集成com.ar-optimization.scheduler帧级资源调度器含热区管理、预测加载com.ar-optimization.anchor-governor锚点治理套件含健康度评估、分时回收com.ar-optimization.sensor-pipeline双线程传感器管道含C Plugin、RingBuffer、校准器集成只需三步在Package Manager中添加Git URL在AR Session GameObject上添加AROptimizerController组件在Inspector中勾选启用的策略支持混合启用如只开调度锚点治理。更关键的是CI/CD集成。我们在Jenkins Pipeline中加入了AR性能门禁AR Performance Gatestage(AR Performance Test) { steps { script { // 运行自动化测试脚本采集CEI指标 sh python3 ar_test_runner.py --appbuilds/ARApp.apk --deviceHUAWEI-MATE40 // 检查CEI是否≥0.85否则失败 if (readFile(cei_result.txt).toFloat() 0.85) { error AR CEI too low: ${readFile(cei_result.txt)} } } } }这个门禁让性能回归问题在PR阶段就被拦截避免劣质代码合入主干。目前该Package已在公司内部12个AR项目中落地平均节省QA性能回归工时63%。5.3 跨平台适配iOS与Windows Mixed Reality的差异化处理本文策略虽以Android/AR Foundation为主但已验证在iOSARKit和Windows Mixed RealityOpenXR上的可行性。差异点在于iOS端ARKit的ARFrame.anchors访问本身是线程安全的无需双线程解耦但帧级调度和锚点治理同样有效。唯一调整是将C Plugin替换为Objective-C Wrapper调用[ARFrame getAnchors]。Windows MR端OpenXR的xrWaitFrame()是阻塞调用必须放入独立线程。我们复用双线程架构但RingBuffer改为Windows Event Shared Memory实现校准逻辑不变。共性原则所有平台都遵循“传感器采集与渲染分离”这一核心思想只是底层API不同。我们提供Platform Abstraction LayerPALC#层代码95%复用。实测表明在iPhone 12A14上三策略组合使CEI从0.651提升至0.88335.6%在HP Reverb G2Intel i7-10700K上从0.592提升至0.84142.0%。这证明策略的普适性——它解决的是AR应用的共性矛盾而非特定平台的临时补丁。6. 我的实际经验那些文档里不会写的细节最后分享几个血泪教训都是我在凌晨三点调试时记下的第一不要相信ARFoundation的“自动销毁”。哪怕你用了最新的ARAnchorManager只要项目里存在任何ARSession.nativeSession的直接调用Native Anchor就可能泄漏。我的做法是全局搜索nativeSession全部替换成ARFoundation封装API并在OnApplicationPause(true)时强制调用anchorManager.DestroyAllAnchors()——这招在后台切回前台时救了我三次。第二帧级调度的预测半径不是越大越好。我们最初设为5米结果在狭小房间内预测范围覆盖整个空间导致所有资源全加载内存爆炸。后来发现2.5米是黄金值它覆盖了人眼自然聚焦范围人眼舒适视距约2~3米且在大多数室内场景中用户转头超过2.5米时原视野已基本退出关注区。第三双线程解耦后务必关闭Unity的VSync。因为传感器线程提供的时间戳是绝对的而VSync会强制渲染帧对齐显示器刷新率造成时间戳与实际渲染时刻错位。我们在PlayerSettings中设置Application.targetFrameRate 60并关闭QualitySettings.vSyncCount改用Time.captureFramerate 60做软同步效果更稳。第四也是最重要的优化永远服务于体验而非参数。有次我把CEI刷到了0.92但用户反馈“虚拟按钮太灵敏老是误触”。回头一看是双线程解耦后触摸响应延迟从80ms降到12ms手指还没抬起来点击事件已触发两次。于是我在输入层加了15ms的防抖CEI微降0.003但NPS净推荐值从32升到67。记住数字是工具人才是终点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2639805.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;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…