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

Furion定时任务UI管理界面怎么玩?/myjob路径配置与动态任务增删改查实战

Furion定时任务UI管理界面实战指南从配置到动态任务管理在.NET生态系统中定时任务管理一直是开发者需要面对的基础设施挑战之一。传统方式下我们往往需要依赖Windows任务计划程序或第三方服务不仅部署复杂还缺乏实时监控能力。Furion框架提供的ScheduleUI功能模块彻底改变了这一局面它将定时任务的管理和监控集成到了Web界面中让开发者能够像管理普通页面一样管理后台任务。1. 环境准备与基础配置让我们从最基础的配置开始逐步构建一个完整的定时任务管理系统。首先确保你已经创建了一个基于.NET 6的Web项目并安装了Furion.Pure 4.8.8.48或更高版本。在Program.cs中我们需要进行以下基础配置var builder WebApplication.CreateBuilder(args).Inject(); builder.Services.AddControllers().AddInject(); builder.Services.AddSchedule(); // 添加定时任务服务 var app builder.Build(); // 配置ScheduleUI界面 app.UseScheduleUI(options { options.RequestPath /myjob; // 自定义访问路径 options.DisableOnProduction false; // 生产环境也启用 }); app.UseInject(string.Empty); app.MapControllers(); app.Run();这段代码做了几件重要的事情通过AddSchedule()添加了定时任务服务使用UseScheduleUI()启用了UI管理界面将UI界面的访问路径设置为/myjob你可以更改为任何你喜欢的路径关键配置参数说明参数名类型默认值说明RequestPathstring/scheduleUI界面的访问路径DisableOnProductionbooltrue是否在生产环境禁用UIDisplayNamestring定时任务UI界面显示的名称Themestringdefault界面主题风格配置完成后启动项目并访问/myjob路径你将看到一个简洁的任务管理界面。虽然此时还没有任何任务显示但这个界面已经具备了监控和管理任务的所有基础功能。2. 创建与配置定时任务有了基础环境后我们来创建第一个定时任务。Furion中的定时任务是通过实现IJob接口来定义的。using Furion.Schedule; public class LogCleanJob : IJob { private readonly ILoggerLogCleanJob _logger; public LogCleanJob(ILoggerLogCleanJob logger) { _logger logger; } public Task ExecuteAsync(JobExecutingContext context, CancellationToken stoppingToken) { _logger.LogInformation($正在执行日志清理任务任务ID{context.JobDetail.JobId}); // 实际的日志清理逻辑放在这里 return Task.CompletedTask; } }这个简单的LogCleanJob定义了一个日志清理任务。接下来我们需要在服务配置中注册这个任务builder.Services.AddSchedule(options { options.AddJobLogCleanJob(log_clean, Triggers.Daily(hour: 2)); // 每天凌晨2点执行 });Furion提供了多种触发器类型可以满足不同场景的需求立即执行Triggers.Now()周期性执行Triggers.PeriodSeconds(30)每30秒每日执行Triggers.Daily(hour: 14, minute: 30)每天14:30每周执行Triggers.Weekly(DayOfWeek.Monday, hour: 9)每周一9点CRON表达式Triggers.Cron(0 0 12 * * ?)每天中午12点在UI界面中你可以看到任务的详细信息任务名称和描述上次执行时间下次执行时间执行状态运行中、等待、暂停等执行次数统计错误次数统计3. 动态任务管理实战静态配置的任务虽然简单但在实际应用中我们经常需要根据业务需求动态地添加、修改或删除任务。Furion通过ISchedulerFactory接口提供了完整的动态任务管理能力。首先我们创建一个控制器来处理动态任务操作[DynamicApiController] public class JobController { private readonly ISchedulerFactory _schedulerFactory; public JobController(ISchedulerFactory schedulerFactory) { _schedulerFactory schedulerFactory; } // 添加动态任务 [HttpPost] public string AddDynamicJob([FromBody] DynamicJobRequest request) { var jobId $dynamic_{Guid.NewGuid():N}; // 根据请求参数创建不同类型的触发器 ITrigger trigger request.TriggerType switch { interval Triggers.PeriodSeconds(request.Interval), daily Triggers.Daily(hour: request.Hour, minute: request.Minute), cron Triggers.Cron(request.CronExpression), _ throw new ArgumentException(不支持的触发器类型) }; _schedulerFactory.AddJobDynamicJob(jobId, trigger); return jobId; } // 暂停任务 [HttpPost({jobId}/pause)] public bool PauseJob(string jobId) { return _schedulerFactory.TryPauseJob(jobId); } // 恢复任务 [HttpPost({jobId}/resume)] public bool ResumeJob(string jobId) { return _schedulerFactory.TryResumeJob(jobId); } // 删除任务 [HttpDelete({jobId})] public bool RemoveJob(string jobId) { return _schedulerFactory.TryRemoveJob(jobId); } // 获取所有任务 [HttpGet] public IEnumerableJobInfo GetAllJobs() { return _schedulerFactory.GetJobs() .Select(j new JobInfo { JobId j.JobId, Status j.Status.ToString(), NextRunTime j.NextRunTime, LastRunTime j.LastRunTime }); } } public class DynamicJobRequest { public string TriggerType { get; set; } public int Interval { get; set; } public int Hour { get; set; } public int Minute { get; set; } public string CronExpression { get; set; } } public class JobInfo { public string JobId { get; set; } public string Status { get; set; } public DateTime? NextRunTime { get; set; } public DateTime? LastRunTime { get; set; } }这个控制器提供了完整的任务生命周期管理功能添加任务通过POST请求可以动态创建各种类型的定时任务暂停/恢复任务无需删除任务可以临时暂停后再恢复删除任务彻底移除不再需要的任务查询任务获取当前所有任务的状态信息提示动态任务的修改操作通常通过删除新增组合实现因为直接修改现有任务的触发器可能会带来意想不到的副作用。4. 任务持久化与高可用设计在生产环境中应用可能会重启或需要多实例部署这就要求定时任务的状态能够持久化并在多个实例之间保持一致性。Furion通过IJobPersistence接口提供了任务持久化的扩展点。首先我们需要定义数据库模型来存储任务信息[SugarTable(Jobs)] public class JobModel { [SugarColumn(IsPrimaryKey true)] public string JobId { get; set; } public string GroupName { get; set; } public string JobType { get; set; } public string AssemblyName { get; set; } public string Description { get; set; } public bool Concurrent { get; set; } public bool IncludeAnnotations { get; set; } public string Properties { get; set; } public DateTime UpdatedTime { get; set; } public bool IsDelete { get; set; } } [SugarTable(JobTriggers)] public class JobTriggerModel { [SugarColumn(IsPrimaryKey true)] public string TriggerId { get; set; } public string JobId { get; set; } public string TriggerType { get; set; } public string AssemblyName { get; set; } public string Args { get; set; } public string Description { get; set; } public TriggerStatus Status { get; set; } public DateTime? StartTime { get; set; } public DateTime? EndTime { get; set; } public DateTime? LastRunTime { get; set; } public DateTime? NextRunTime { get; set; } public long NumberOfRuns { get; set; } public long MaxNumberOfRuns { get; set; } public int NumberOfErrors { get; set; } public int MaxNumberOfErrors { get; set; } public DateTime UpdatedTime { get; set; } public bool IsDelete { get; set; } }接下来实现IJobPersistence接口public class DatabaseJobPersistence : IJobPersistence { private readonly ISqlSugarClient _db; public DatabaseJobPersistence(ISqlSugarClient db) { _db db; } public void OnChanged(PersistenceContext context) { var job JsonConvert.DeserializeObjectJobModel(context.ConvertToJSON()); job.UpdatedTime DateTime.Now; switch (context.Behavior) { case PersistenceBehavior.Appended: case PersistenceBehavior.Updated: job.IsDelete false; _db.Storageable(job).ExecuteCommand(); break; case PersistenceBehavior.Removed: job.IsDelete true; _db.Updateable(job).ExecuteCommand(); break; } } public SchedulerBuilder OnLoading(SchedulerBuilder builder) { return builder; } public void OnTriggerChanged(PersistenceTriggerContext context) { var trigger JsonConvert.DeserializeObjectJobTriggerModel(context.ConvertToJSON()); trigger.UpdatedTime DateTime.Now; switch (context.Behavior) { case PersistenceBehavior.Appended: case PersistenceBehavior.Updated: trigger.IsDelete false; _db.Storageable(trigger).ExecuteCommand(); break; case PersistenceBehavior.Removed: trigger.IsDelete true; _db.Updateable(trigger).ExecuteCommand(); break; } } public IEnumerableSchedulerBuilder Preload() { var allJobs new ListSchedulerBuilder(); // 加载静态任务 var staticJobs App.EffectiveTypes.ScanToBuilders(); allJobs.AddRange(staticJobs); // 加载数据库中的动态任务 var dbJobs _db.QueryableJobModel() .Where(j !j.IsDelete) .ToList(); var dbTriggers _db.QueryableJobTriggerModel() .Where(t !t.IsDelete) .ToList(); foreach (var job in dbJobs) { // 跳过已经在静态任务中定义的 if (staticJobs.Any(s s.GetJobBuilder().JobId job.JobId)) continue; var jobBuilder JobBuilder.Create(job.JobId).LoadFrom(job); var triggers dbTriggers .Where(t t.JobId job.JobId) .Select(t TriggerBuilder.Create(t.TriggerId).LoadFrom(t)) .ToArray(); allJobs.Add(SchedulerBuilder.Create(jobBuilder, triggers)); } return allJobs; } }最后在Program.cs中注册持久化服务// 配置数据库连接 var db new SqlSugarClient(new ConnectionConfig { ConnectionString 你的数据库连接字符串, DbType DbType.SqlServer, IsAutoCloseConnection true }); // 初始化数据库表 db.CodeFirst.InitTablesJobModel, JobTriggerModel(); builder.Services.AddSingletonISqlSugarClient(db); builder.Services.AddSchedule(options { options.AddPersistenceDatabaseJobPersistence(); });这种持久化方案带来了几个重要优势任务状态持久化即使应用重启所有任务状态都能恢复多实例支持多个应用实例可以共享同一个任务状态历史记录可以追踪任务的执行历史和状态变化动态任务持久化动态创建的任务也会被保存不会丢失5. 高级功能与最佳实践掌握了基础功能后让我们来看一些高级用法和最佳实践这些技巧可以帮助你构建更健壮、更易维护的定时任务系统。5.1 任务异常处理与重试机制定时任务在执行过程中可能会遇到各种异常情况良好的错误处理机制至关重要。public class RobustJob : IJob { public Task ExecuteAsync(JobExecutingContext context, CancellationToken stoppingToken) { try { // 可能会失败的业务逻辑 ProcessData(); } catch (Exception ex) { context.JobDetail.JobLogger?.LogError(ex, 任务执行失败); // 决定是否重试 if (context.Trigger.NumberOfErrors context.Trigger.MaxNumberOfErrors) { context.SetRetry(); // 标记为需要重试 } // 将异常信息保存到任务结果中 context.Result $失败{ex.Message}; context.ElapsedTime context.Stopwatch.ElapsedMilliseconds; throw; // 重新抛出异常以触发重试机制 } return Task.CompletedTask; } }在任务配置时可以设置重试参数builder.Services.AddSchedule(options { options.AddJobRobustJob(robust_job, Triggers.PeriodSeconds(30) .SetNumRetries(3) // 最大重试次数 .SetRetryTimeout(5000)); // 重试间隔(毫秒) });5.2 任务依赖与顺序执行有时我们需要确保某些任务按特定顺序执行或者一个任务依赖于另一个任务的结果。Furion虽然没有内置的依赖管理但我们可以通过任务状态来实现类似功能。public class DependentJob : IJob { private readonly ISchedulerFactory _scheduler; public DependentJob(ISchedulerFactory scheduler) { _scheduler scheduler; } public Task ExecuteAsync(JobExecutingContext context, CancellationToken stoppingToken) { // 检查前置任务是否完成 var prerequisiteJob _scheduler.GetJob(prerequisite_job); if (prerequisiteJob?.Status ! JobStatus.COMPLETED) { context.Result 等待前置任务完成; return Task.CompletedTask; // 本次不执行实际逻辑 } // 实际业务逻辑 ExecuteBusinessLogic(); // 触发后续任务 _scheduler.ResumeJob(next_job); return Task.CompletedTask; } }5.3 任务性能监控对于关键任务监控其执行性能和资源消耗非常重要。我们可以通过自定义拦截器来实现public class MonitoringJobInterceptor : IJobInterceptor { private readonly ILoggerMonitoringJobInterceptor _logger; public MonitoringJobInterceptor(ILoggerMonitoringJobInterceptor logger) { _logger logger; } public Task OnExecutiongAsync(JobExecutingContext context, CancellationToken stoppingToken) { _logger.LogInformation($任务[{context.JobDetail.JobId}]开始执行); context.Properties[StartTime] DateTime.Now; context.Properties[MemoryBefore] GC.GetTotalMemory(false); return Task.CompletedTask; } public Task OnExecutedAsync(JobExecutedContext context, CancellationToken stoppingToken) { var startTime (DateTime)context.Properties[StartTime]; var memoryBefore (long)context.Properties[MemoryBefore]; var memoryAfter GC.GetTotalMemory(false); _logger.LogInformation($任务[{context.JobDetail.JobId}]执行完成耗时{(DateTime.Now - startTime).TotalMilliseconds}ms内存变化{memoryAfter - memoryBefore}bytes); // 记录性能指标到数据库或监控系统 RecordPerformanceMetrics( context.JobDetail.JobId, DateTime.Now - startTime, memoryAfter - memoryBefore, context.Exception ! null); return Task.CompletedTask; } }注册拦截器builder.Services.AddSchedule(options { options.AddJobCriticalJob(critical_job, Triggers.Hourly()) .AddInterceptorMonitoringJobInterceptor(); });5.4 UI界面自定义Furion的ScheduleUI界面虽然开箱即用但有时我们需要根据业务需求进行一些自定义。app.UseScheduleUI(options { options.RequestPath /admin/jobs; options.DisplayName 后台任务管理; options.Theme dark; // 添加自定义CSS和JS options.AddCustomResource(/css/custom-schedule.css); options.AddCustomResource(/js/custom-schedule.js); // 控制列的显示/隐藏 options.ConfigureColumnVisibility(visible: [jobId, status, nextRunTime], hidden: [assemblyName, description]); // 添加自定义操作按钮 options.AddCustomAction(手动执行, jobId $executeJob({jobId})); });通过这些高级功能你可以构建出适应复杂业务场景的定时任务系统同时保持良好的可维护性和可观测性。

相关文章:

Furion定时任务UI管理界面怎么玩?/myjob路径配置与动态任务增删改查实战

Furion定时任务UI管理界面实战指南:从配置到动态任务管理 在.NET生态系统中,定时任务管理一直是开发者需要面对的基础设施挑战之一。传统方式下,我们往往需要依赖Windows任务计划程序或第三方服务,不仅部署复杂,还缺乏…...

别再死磕源码了!用Live555 MediaServer快速搭建你的第一个RTSP流媒体服务(Windows/Linux保姆级教程)

别再死磕源码了!用Live555 MediaServer快速搭建你的第一个RTSP流媒体服务(Windows/Linux保姆级教程) 流媒体技术正逐渐渗透到视频监控、在线教育、视频会议等各个领域,而RTSP(Real Time Streaming Protocol&#xff09…...

Kettle调度避坑实录:从.bat脚本编写到Windows任务计划,我踩过的那些雷

Kettle调度避坑实录:从.bat脚本编写到Windows任务计划,我踩过的那些雷 第一次尝试用Windows任务计划调度Kettle作业时,我以为按照教程一步步操作就能轻松搞定。直到凌晨三点还在排查为什么任务计划显示"成功执行",但数据…...

让你的UI“动”起来:在Unity Canvas上完美融合粒子特效的两种实用方法

让UI与粒子特效完美共舞:Unity Canvas特效融合实战指南 在游戏界面设计中,UI与粒子特效的融合往往能创造出令人惊艳的视觉效果。想象一下,当玩家点击按钮时迸发的火花,或是菜单界面中流动的光效,这些动态元素能为静态界…...

DCNv4深度解析:高效可变形卷积的技术实现与架构设计

DCNv4深度解析:高效可变形卷积的技术实现与架构设计 【免费下载链接】DCNv4 [CVPR 2024] Deformable Convolution v4 项目地址: https://gitcode.com/gh_mirrors/dc/DCNv4 DCNv4(Deformable Convolution v4)是OpenGVLab发布的最新可变…...

Navicat无限试用终极教程:macOS用户告别14天限制的完整指南

Navicat无限试用终极教程:macOS用户告别14天限制的完整指南 【免费下载链接】navicat_reset_mac navicat mac版无限重置试用期脚本 Navicat Mac Version Unlimited Trial Reset Script 项目地址: https://gitcode.com/gh_mirrors/na/navicat_reset_mac 还在为…...

3个技术要点让Cesium风场可视化从概念到实现

3个技术要点让Cesium风场可视化从概念到实现 【免费下载链接】cesium-wind wind layer of cesium 项目地址: https://gitcode.com/gh_mirrors/ce/cesium-wind Cesium-Wind是一款专为Cesium三维地球引擎设计的风场可视化插件,它将抽象的气象数据转化为动态的粒…...

终极SQL血缘分析工具:sqllineage让数据流向一目了然

终极SQL血缘分析工具:sqllineage让数据流向一目了然 【免费下载链接】sqllineage SQL Lineage Analysis Tool powered by Python 项目地址: https://gitcode.com/gh_mirrors/sq/sqllineage 你是否曾在复杂的ETL管道中迷失方向,不知道某个数据字段…...

ArchivePasswordTestTool:终极免费压缩包密码恢复工具完整指南

ArchivePasswordTestTool:终极免费压缩包密码恢复工具完整指南 【免费下载链接】ArchivePasswordTestTool 利用7zip测试压缩包的功能 对加密压缩包进行自动化测试密码 项目地址: https://gitcode.com/gh_mirrors/ar/ArchivePasswordTestTool 你是否曾经面对一…...

如何在群晖NAS上打造个人百度云管家?三步解锁云端文件同步新体验

如何在群晖NAS上打造个人百度云管家?三步解锁云端文件同步新体验 【免费下载链接】synology-baiduNetdisk-package 项目地址: https://gitcode.com/gh_mirrors/sy/synology-baiduNetdisk-package 你是否曾经为群晖NAS和百度云之间的文件同步而烦恼&#xff…...

探索三维互联网:Firefox Reality如何重新定义VR/AR浏览体验

探索三维互联网:Firefox Reality如何重新定义VR/AR浏览体验 【免费下载链接】FirefoxReality A fast and secure browser for standalone virtual-reality and augmented-reality headsets. 项目地址: https://gitcode.com/gh_mirrors/fi/FirefoxReality 想象…...

FPGA设计避坑指南:为什么你的Mealy状态机输出有毛刺?输出寄存实战解析

FPGA设计避坑指南:为什么你的Mealy状态机输出有毛刺?输出寄存实战解析 在高速FPGA设计中,状态机的稳定性往往决定着整个系统的可靠性。最近调试一个千兆以太网控制器时,我遇到了一个诡异的现象——状态机输出的控制信号偶尔会出现…...

Arm GICv3中断控制器架构与关键寄存器解析

1. Arm GICv3中断控制器架构概述在现代处理器架构中,中断控制器是连接外设与CPU核心的关键枢纽。Arm的通用中断控制器(Generic Interrupt Controller, GIC)经过多代演进,GICv3架构引入了对64位系统的全面支持,并通过系统寄存器接口提供了更灵…...

阿里云盘API凭证获取机制深度解析与安全集成方案

阿里云盘API凭证获取机制深度解析与安全集成方案 【免费下载链接】aliyundriver-refresh-token QR Code扫码获取阿里云盘refresh token For Web 项目地址: https://gitcode.com/gh_mirrors/al/aliyundriver-refresh-token 在云存储API集成开发中,阿里云盘Ref…...

强化学习在智能决策系统中的应用与实践

1. 项目背景与核心价值去年在开发一个智能决策系统时,我发现传统规则引擎在面对复杂动态环境时表现乏力。这促使我开始探索强化学习(RL)在自主推理领域的应用可能性。经过半年多的实践验证,这种结合方式在动态路径规划、资源调度等…...

终极风扇控制指南:如何用FanControl让电脑静音又高效散热

终极风扇控制指南:如何用FanControl让电脑静音又高效散热 【免费下载链接】FanControl.Releases This is the release repository for Fan Control, a highly customizable fan controlling software for Windows. 项目地址: https://gitcode.com/GitHub_Trending…...

SRWE终极指南:简单运行时窗口编辑器的完整使用教程

SRWE终极指南:简单运行时窗口编辑器的完整使用教程 【免费下载链接】SRWE Simple Runtime Window Editor 项目地址: https://gitcode.com/gh_mirrors/sr/SRWE 你是否曾遇到过游戏截图分辨率不够高、应用程序窗口布局不灵活、或者需要为不同平台调整窗口尺寸的…...

3步解决Windows苹果设备连接难题:告别iTunes臃肿安装的轻量方案

3步解决Windows苹果设备连接难题:告别iTunes臃肿安装的轻量方案 【免费下载链接】Apple-Mobile-Drivers-Installer Powershell script to easily install Apple USB and Mobile Device Ethernet (USB Tethering) drivers on Windows! 项目地址: https://gitcode.c…...

别再傻傻分不清了!用Python和R语言代码实例,5分钟搞懂PDF和CDF的区别

别再傻傻分不清了!用Python和R语言代码实例,5分钟搞懂PDF和CDF的区别 第一次接触概率密度函数(PDF)和累积分布函数(CDF)时,很多人都会被这两个看似相似实则完全不同的概念搞得晕头转向。作为一名…...

同济高数第七版第一章:函数与极限,我用Python可视化帮你彻底搞懂(附代码)

用Python可视化拆解高数核心概念:从函数到极限的编程实践 数学从来不是纸面上的抽象符号,而是理解世界的语言。当同济大学《高等数学》第七版中的函数曲线在Matplotlib中动态呈现,当ε-δ定义通过动画逐帧展示,理工科学生第一次感…...

从数据清洗到模型输入:深度解析PyTorch中cat、stack、split的工程化使用场景

从数据清洗到模型输入:深度解析PyTorch中cat、stack、split的工程化使用场景 在深度学习项目的实际开发中,数据处理流水线往往占据了整个项目70%以上的工作量。而Tensor的拼接与拆分操作,正是构建高效数据流的关键枢纽。不同于教科书式的API讲…...

高效实战:深度解析cri-dockerd架构与Kubernetes容器运行时最佳实践

高效实战:深度解析cri-dockerd架构与Kubernetes容器运行时最佳实践 【免费下载链接】cri-dockerd dockerd as a compliant Container Runtime Interface for Kubernetes 项目地址: https://gitcode.com/gh_mirrors/cr/cri-dockerd cri-dockerd是一个专业的容…...

孤舟笔记 并发篇一 面试总问AQS,它到底是个啥?凭什么它是并发编程的灵魂

文章目录一、先说结论:AQS 是一把"排队管理器"二、没有 AQS 的世界:每个锁都得自己造轮子三、AQS 的两大核心:state 和队列1. state——一把万能计数器2. CLH 队列——抢不到就排队四、AQS 怎么用?抢锁和释放就两步独占…...

【缺陷检测】基于k-means分割Otsu阈值检测水果和蔬菜缺陷(外部和内部缺陷)附Matlab代码

✅作者简介:热爱科研的Matlab仿真开发者,擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。🍎 往期回顾关注个人主页:Matlab科研工作室👇 关注我领取海量matlab电子书和…...

别再死记硬背了!用Python递归函数5分钟搞定二叉树前序/中序/后序转换(附PTA真题解析)

用Python递归思维破解二叉树遍历转换难题 第一次接触二叉树的前序、中序、后序遍历转换时,你是否也曾在各种递归调用和数组下标中迷失方向?作为数据结构学习路上的经典难题,这三种遍历方式的相互转换常常让初学者感到头疼。但今天我要分享的&…...

基于AI与事件驱动的临床安全网系统:从概念到2.5小时原型实践

1. 项目概述:一个在2.5小时内诞生的临床安全网原型 在初级医疗领域,全科医生(GP)每天都会重复成百上千次同一句医嘱:“如果情况没有好转,请回来复诊。”这句话在医学上被称为“安全网”(Safety …...

打卡信奥刷题(3190)用C++实现信奥题 P8085 [COCI 2011/2012 #4] KRIPTOGRAM

P8085 [COCI 2011/2012 #4] KRIPTOGRAM 题目描述 现有一段明文和一部分密文。明文和密文都由英文单词组成,且密文中的一个单词必然对应着明文中的一个单词。 求给出的密文在明文中可能出现的最早位置。 输入格式 第一行,若干个英文单词和一个 $&…...

KiCad设计开源Snapdragon 845载板:高性能边缘计算实战

1. 开源硬件新标杆:基于KiCad的Snapdragon 845载板设计解析 当大多数商用开发板还在使用闭源EDA工具时,Antmicro团队用KiCad完成了一次漂亮的示范——他们为Quectel SA800U-WF模块设计的开源载板,不仅完整释放了骁龙845处理器的潜力&#xff…...

iMX93 Pro工业开发套件:边缘AI与实时控制解析

1. VOIPAC iMX93 Pro工业级开发套件深度解析作为一名长期跟踪嵌入式开发板的技术博主,我最近详细研究了VOIPAC公司推出的iMX93 Pro工业级开发套件。这款基于NXP i.MX 93处理器的开发平台,在边缘AI和工业自动化领域展现出独特优势。与常见的树莓派或Jetso…...

终极指南:如何在Windows上直接安装安卓应用?APK安装器完整教程

终极指南:如何在Windows上直接安装安卓应用?APK安装器完整教程 【免费下载链接】APK-Installer An Android Application Installer for Windows 项目地址: https://gitcode.com/GitHub_Trending/ap/APK-Installer 想在Windows电脑上直接运行手机应…...