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

从数据采集到回放验证:ADTF 适配 ROS 的 ADAS 测试实践佑

一、简化查询

1. 先看一下查询的例子

///

/// 账户获取服务

///

///

///

public class AccountGetService(AccountTable table, IShadowBuilder builder)

{

private readonly SqlSource _source = new(builder.DataSource);

private readonly IParamQuery _accountQuery

= builder.BuildResult(

table.ToQuery()

.And(account => account.Id.Equal())

.ToSelect()

.SelectSelfColumns()

);

///

/// 获取账户

///

///

///

///

public Task GetAsync(Account param, CancellationToken token = default)

=> _accountQuery.GetFirstAsync(_source, param, token);

}

2. 调用方式如下

var account = await accountGetService.GetAsync(new Account { Id = 1L });

3. 这个例子我们可以简化一下

参数直接使用Id(类型由Account简化为long)

很大一部分数据库操作都只有一个参数(GetById、GetByName等)

为此定义一个含单个属性类有点浪费

按上面的例子使用实体类作为参数也感觉怪怪的

DBShadow.net支持这种简化操作

单个参数无需封装,直接使用参数类型即可

public class AccountGetService(AccountTable table, IShadowBuilder builder)

{

private readonly SqlSource _source = new(builder.DataSource);

private readonly IParamQuery _accountQuery

= builder.BuildResult(

table.ToQuery()

.And(account => account.Id.Equal())

.ToSelect()

.SelectSelfColumns()

);

public Task GetAsync(long accountId, CancellationToken token = default)

=> _accountQuery.GetFirstAsync(_source, param, token);

}

var account = await accountGetService.GetAsync(1L);

4. 特别注意不要在Dapper中这样使用

Dapper是不支持这种简化操作

以下Dapper错误示例会抛出异常

System.InvalidOperationException:“Must add values for the following parameters: @Id”

await using var conn = _dataSource.CreateConnection();

var sql = "SELECT \"Id\",\"Title\",\"Content\",\"Done\",\"LastTime\" FROM \"Todo\" WHERE \"Id\"=@Id";

var first = await conn.QueryFirstOrDefaultAsync(sql, 1L);

二、 集合参数也支持简化操作

1. IN查询简化的例子

In的参数名是可选和前面例子的Equal是一样的,默认字段名

In查询的实参支持数组、集合及字典

只有In一个集合参数时支持直接传数组、集合或字典

这时参数名(eg: AccountIds)就无所谓取什么名了

因为不需要按参数名反射获取属性了

///

/// 批量账户获取服务

///

///

///

public class AccountBatchService(AccountTable table, IShadowBuilder builder)

{

private readonly SqlSource _source = new(builder.DataSource);

private readonly IParamQuery _accountQuery

= builder.BuildResult(

table.ToQuery()

.And(account => account.Id.In("AccountIds"))

.ToSelect()

.SelectSelfColumns()

);

///

/// 批量获取账户

///

///

///

///

public IAsyncEnumerable GetAsync(long[] accountIds, CancellationToken token = default)

=> _accountQuery.QueryAsync(_source, accountIds, token);

}

2. 以上调用的例子如下

[Fact]

public async Task Batch()

{

var count = 0;

var service = new AccountBatchService(table, builder);

var list = service.GetAsync([1L, 2L, 3L]);

await foreach (var item in list)

{

_output.WriteLine($"{item.Id}:{item.Amount}");

count++;

}

Assert.Equal(3, count);

}

// 1:100

// 2:200

// 3:300

三、泛型查询

1. 泛型服务类代码

该服务用来按字段Id查询表Account

参数TParam是泛型,这样可以直接使用DTO参数来查询,减少类型转化的开销

返回值TAccount也是泛型,这样就可以直接返回视图模型或者领域模型

这样的泛型服务类就非常的通用

///

/// 账户获取泛型服务

///

///

///

public class AccountGetService(AccountTable table, IShadowBuilder builder)

{

private readonly SqlSource _source = new(builder.DataSource);

private readonly IParamQuery _accountQuery

= builder.BuildResult(

table.ToQuery()

.And(account => account.Id.Equal())

.ToSelect()

.SelectSelfColumns()

);

///

/// 获取账户

///

///

///

///

public Task GetAsync(TParam param, CancellationToken token = default)

=> _accountQuery.GetFirstAsync(_source, param, token);

}

2. 中规中矩的调用方式

var service = new AccountGetService(_table, _builder);

var account = await service.GetAsync(new Account { Id = 1L });

Assert.NotNull(account);

3. 支持简单调用方式

var service = new AccountGetService(_table, _builder);

var account = await service.GetAsync(1L);

Assert.NotNull(account);

四、总结

DBShadow.net预编译比较智能

只有1个参数时支持化繁为简,支持直接传值做为参数值

这样可以节约定义只有一个属性的参数类

参数和返回值类型还可以定义为泛型,可以做到更加灵活闹笛似旧

http://www.jsqmd.com/news/619910/

相关文章:

  • SpringBoot3与OAuth2.1实战:从零搭建授权服务器
  • 专业字体优化指南:3步完成屏幕阅读字体配置,告别视觉疲劳
  • 同一个需求,我先出技术方案,再让AI出方案——差距让我沉默了倭
  • OpenClaw+Qwen3-4B组合技能:多模块协作自动化
  • 野火imx6ull上跑SOEM主站:从编译到点亮LED的完整避坑指南
  • claw-code 源码分析:成本追踪(Cost)与 Hook——企业落地时,计量与策略注入该挂在哪一层?
  • uni-app怎么实现App沉浸式导航栏 uni-app透明标题栏配置【详解】
  • 喜欢搞技术的高技术、喜欢搞业务的搞业务
  • WindRunnerMax窒
  • 高效直链文件分享平台深度评测(二)
  • 项目过程域--客户验收过程说明
  • Wan2.2-I2V-A14B API服务部署教程:Python调用批量生成视频接口
  • [前端 | 布局示例]
  • 3步掌控:钉钉防撤回与多开工具的终极使用指南
  • HE693RTD665A输入模块
  • 紧急预警!2025年起Java 8/Oracle Forms系统将丧失AI集成资质——30天迁移倒计时应对方案
  • 详细解析Spring如何解决循环依赖问题依
  • hive介绍
  • 基于模型预测控制(自带的mpc模块)和最优控制理论的Carsim与Matlab/simulin...
  • 从A*到Theta*:探索任意角度路径规划的演进与实战
  • 答辩AI工具盘点:10款高效选择(含aibiye)与模板使用经验。
  • C# 面试高频题:装箱和拆箱是如何影响性能的?跋
  • QT点云渲染实战--从QGLWidget到交互式3D可视化
  • Vitis HLS 2021.2 配置 OpenCV 完整避坑指南:从CMake编译到环境变量设置
  • LabelImg标注数据转换全攻略:XML与TXT互转的Python脚本详解
  • 把手工发版产品化:一键部署闭环的工程落地复盘(FastAPI + Paramiko)
  • OpenClaw多模型切换:Qwen3-14b_int4_awq与其他本地模型的协同使用
  • 02_TiDB向量搜索深度实战:从基础概念到生产部署
  • IDEA中模块位置创建错了想重建时提示改模块已存在的解决办法
  • claw-code 源码分析:结构化输出与重试——`structured_output` 一类开关如何改变「可解析性」与失败语义?