news 2026/4/30 1:08:24

《上海市卫生健康”信息技术应用创新”白皮书》C#/.NET “A组件”认定事件始末与最终结果

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
《上海市卫生健康”信息技术应用创新”白皮书》C#/.NET “A组件”认定事件始末与最终结果

一、简化查询

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个参数时支持化繁为简,支持直接传值做为参数值

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

参数和返回值类型还可以定义为泛型,可以做到更加灵活疵仝阂写

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/30 1:05:28

Linux 5.15 LTS内核新特性与优化解析

1. Linux 5.15 LTS版本概览Linux内核5.15长期支持版(LTS)于2021年10月31日正式发布,这是Linus Torvalds领导下的第21个LTS内核版本。作为一名长期跟踪内核开发的系统工程师,我认为这个版本虽然更新规模中等,但在文件系…

作者头像 李华
网站建设 2026/4/30 1:05:25

TC8测试实战:用Python脚本自动化验证SOME/IP服务发现与RPC(附代码片段)

TC8测试实战:用Python脚本自动化验证SOME/IP服务发现与RPC 在汽车电子系统从传统总线向以太网架构迁移的浪潮中,SOME/IP协议凭借其服务导向特性成为车载通信的核心支柱。但面对TC8测试规范中上百个涉及服务发现、序列化校验的重复性用例,手动…

作者头像 李华
网站建设 2026/4/30 1:01:55

梁高省25cm!“高预应力混杂配筋”HPH构造全解读

在工程以及建筑行业的范围里面,最近“HPH构造”当成了被高度留意的高频率用词。事实上,HPH的全部称呼是“高预应力混杂配筋”(Hybrid Prestressing with High-strength reinforcement),这是一项专门为大跨度空间结构而…

作者头像 李华
网站建设 2026/4/30 1:00:57

【YOLOv11多类别野生动物识别目标检测数据集】

YOLOv11多类别野生动物识别目标检测数据集 📊 数据集基本信息 目标类别: [‘antelope’, ‘badger’, ‘bat’, ‘bear’, ‘bee’, ‘beetle’, ‘bison’, ‘boar’, ‘butterfly’, ‘cat’, ‘caterpillar’, ‘chimpanzee’, ‘cockroach’, ‘cow’…

作者头像 李华
网站建设 2026/4/30 0:59:41

Go语言如何用strings.Builder_Go语言strings.Builder教程【总结】

strings.Builder 写入前必须初始化,否则行为不可靠;正确做法是声明后调用 Reset() 或依赖 Go 1.10 零值(仅限未写入前),复用时必重置,String() 返回拷贝,不共享底层数组。strings.Builder 写入前…

作者头像 李华